Fix mixed spaces/tabs in wscripts
[ardour.git] / libs / backends / wavesaudio / waves_audiobackend.midi.cc
index 94c674d073728ee16cd8f75e5bab1f9a8f11f144..296f3564f9d4a8bb870fda2f8bf475dafbe7f489 100644 (file)
-/*\r
-    Copyright (C) 2014 Waves Audio Ltd.\r
-\r
-    This program is free software; you can redistribute it and/or modify\r
-    it under the terms of the GNU General Public License as published by\r
-    the Free Software Foundation; either version 2 of the License, or\r
-    (at your option) any later version.\r
-\r
-    This program is distributed in the hope that it will be useful,\r
-    but WITHOUT ANY WARRANTY; without even the implied warranty of\r
-    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the\r
-    GNU General Public License for more details.\r
-\r
-    You should have received a copy of the GNU General Public License\r
-    along with this program; if not, write to the Free Software\r
-    Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.\r
-\r
-*/\r
-\r
-#include <boost/assign/list_of.hpp>\r
-\r
-#include "waves_audiobackend.h"\r
-#include "waves_midiport.h"\r
-#include "waves_midi_event.h"\r
-#include "waves_midi_buffer.h"\r
-\r
-using namespace ARDOUR;\r
-\r
-#ifdef __MACOS__\r
-\r
-const std::vector<std::string> WavesAudioBackend::__available_midi_options = boost::assign::list_of ("None") ("CoreMIDI");\r
-\r
-#elif _WINDOWS\r
-\r
-const std::vector<std::string> WavesAudioBackend::__available_midi_options = boost::assign::list_of ("None") ("Multimedia Extensions");\r
-\r
-#endif\r
-\r
-\r
-std::vector<std::string> \r
-WavesAudioBackend::enumerate_midi_options () const\r
-{\r
-    // COMMENTED DBG LOGS */ std::cout << "WavesAudioBackend::enumerate_midi_options ()" << std::endl;\r
-    return __available_midi_options;\r
-}\r
-\r
-\r
-int \r
-WavesAudioBackend::set_midi_option (const std::string& option)\r
-{\r
-    // COMMENTED DBG LOGS */ std::cout << "WavesAudioBackend::set_midi_option ( " << option << " )" << std::endl;\r
-    if (option == __available_midi_options[0]) {\r
-        _use_midi = false;\r
-        // COMMENTED DBG LOGS */ std::cout << "\tNO MIDI system used)" << std::endl;\r
-    }\r
-    else if (option == __available_midi_options[1]) {\r
-        _use_midi = true;\r
-        // COMMENTED DBG LOGS */ std::cout << "\tNO MIDI system used)" << std::endl;\r
-    }\r
-    else {\r
-        std::cerr << "WavesAudioBackend::set_midi_option (): Invalid MIDI option!" << std::endl;\r
-        return -1;\r
-    }\r
-\r
-    return 0;\r
-}\r
-\r
-\r
-std::string\r
-WavesAudioBackend::midi_option () const\r
-{\r
-    // COMMENTED DBG LOGS */ std::cout << "WavesAudioBackend::midi_option ():" << std::endl;\r
-    return * (__available_midi_options.begin () + (_use_midi?1:0));\r
-}\r
-\r
-\r
-int\r
-WavesAudioBackend::midi_event_get (pframes_t& timestamp, size_t& size, uint8_t** buffer, void* port_buffer, uint32_t event_index)\r
-{\r
-    // COMMENTED FREQUENT DBG LOGS */ std::cout << "WavesAudioBackend::midi_event_get ():" << std::endl;\r
-\r
-    if (buffer == NULL) {\r
-        std::cerr << "WavesAudioBackend::midi_event_get () : NULL in the 'buffer' argument!\n";\r
-        return -1;\r
-    }\r
-\r
-    if (port_buffer == NULL) {\r
-        std::cerr << "WavesAudioBackend::midi_event_get () : NULL in the 'port_buffer' argument!\n";\r
-        return -1;\r
-    }\r
-\r
-    WavesMidiBuffer& source = * (WavesMidiBuffer*)port_buffer;\r
-\r
-    if (event_index >= source.size ()) {\r
-        std::cerr << "WavesAudioBackend::midi_event_get () : 'event_index' is out of the number of events stored in 'port_buffer'!\n";\r
-        return -1;\r
-    }\r
-\r
-    WavesMidiEvent* waves_midi_event = source[event_index];\r
-\r
-    timestamp = waves_midi_event->timestamp ();\r
-    size = waves_midi_event->size ();\r
-    *buffer = waves_midi_event->data ();\r
-\r
-    return 0;\r
-}\r
-\r
-\r
-int\r
-WavesAudioBackend::midi_event_put (void* port_buffer, pframes_t timestamp, const uint8_t* buffer, size_t size)\r
-{\r
-    // COMMENTED FREQUENT DBG LOGS */ std::cout << "WavesAudioBackend::midi_event_put ():" << std::endl;\r
-    if (buffer == NULL) {\r
-        std::cerr << "WavesAudioBackend::midi_event_put () : NULL in the 'buffer' argument!\n";\r
-        return -1;\r
-    }\r
-\r
-    if (port_buffer == NULL) {\r
-        std::cerr << "WavesAudioBackend::midi_event_put () : NULL in the 'port_buffer' argument!\n";\r
-        return -1;\r
-    }\r
-\r
-    WavesMidiBuffer& target = * (WavesMidiBuffer*)port_buffer;\r
-    // COMMENTED FREQUENT DBG LOGS */ std::cout << "\t [" << target.name () << "]"<< std::endl;\r
-\r
-    if (target.size () && (pframes_t)target.back ()->timestamp () > timestamp) {\r
-        std::cerr << "WavesAudioBackend::midi_event_put (): The MIDI Event to put is a bit late!" << std::endl;\r
-        std::cerr << "\tprev timestamp is " << (pframes_t)target.back ()->timestamp () << " as the current one is " << timestamp << std::endl;\r
-        return -1;\r
-    }\r
-\r
-    target.push_back (new WavesMidiEvent (timestamp, buffer, size));\r
-    return 0;\r
-}\r
-\r
-\r
-uint32_t\r
-WavesAudioBackend::get_midi_event_count (void* port_buffer)\r
-{\r
-    // COMMENTED FREQUENT DBG LOGS */ std::cout << "WavesAudioBackend::get_midi_event_count (): " << std::endl;\r
-    \r
-    if (port_buffer == NULL) {\r
-        std::cerr << "WavesAudioBackend::get_midi_event_count () : NULL in the 'port_buffer' argument!\n";\r
-        return -1;\r
-    }\r
-\r
-    // COMMENTED FREQUENT DBG LOGS */ std::cout << "\tcount = " << (* (WavesMidiBuffer*)port_buffer).size () << std::endl;\r
-\r
-    return (* (WavesMidiBuffer*)port_buffer).size ();\r
-}\r
-\r
-\r
-void\r
-WavesAudioBackend::midi_clear (void* port_buffer)\r
-{\r
-    // COMMENTED FREQUENT DBG LOGS */ std::cout << "WavesAudioBackend::midi_clear (): " << std::endl;\r
-    if (port_buffer == NULL) {\r
-        std::cerr << "WavesAudioBackend::midi_clear () : NULL in the 'port_buffer' argument!\n";\r
-        return;\r
-    }\r
-\r
-    (* (WavesMidiBuffer*)port_buffer).clear ();\r
-}\r
-\r
-\r
-void\r
-WavesAudioBackend::_changed_midi_devices ()\r
-{\r
-    if (_midi_device_manager.stream (false)) {\r
-        std::cerr << "WavesAudioBackend::_changed_midi_devices (): _midi_device_manager.stream (false) failed!" << std::endl;\r
-        return;\r
-    }\r
-\r
-    _midi_device_manager.stop ();\r
-\r
-    if (_midi_device_manager.start () != 0) {\r
-        std::cerr << "WavesAudioBackend::_changed_midi_devices (): _midi_device_manager.start () failed!" << std::endl;\r
-        return;\r
-    }\r
-\r
-    if (_register_system_midi_ports () != 0) {\r
-        std::cerr << "WavesAudioBackend::_changed_midi_devices (): _register_system_midi_ports () failed!" << std::endl;\r
-        return;\r
-    }\r
-    \r
-    manager.registration_callback ();\r
-\r
-    if (_midi_device_manager.stream (true)) {\r
-        std::cerr << "WavesAudioBackend::_changed_midi_devices (): _midi_device_manager.stream (true) failed!" << std::endl;\r
-        return;\r
-    }\r
-}\r
-\r
-\r
-void\r
-WavesAudioBackend::_unregister_system_midi_ports ()\r
-{\r
-    // COMMENTED DBG LOGS */ std::cout  << "WavesAudioBackend::_unregister_system_midi_ports ()" << std::endl;\r
-    std::vector<WavesMidiPort*> physical_midi_ports = _physical_midi_inputs;\r
-    physical_midi_ports.insert (physical_midi_ports.begin (), _physical_midi_outputs.begin (), _physical_midi_outputs.end ());\r
-\r
-    for (std::vector<WavesMidiPort*>::const_iterator it = physical_midi_ports.begin (); it != physical_midi_ports.end (); ++it) {\r
-        std::vector<WavesDataPort*>::iterator port_iterator = std::find (_ports.begin (), _ports.end (), *it);\r
-        if (port_iterator == _ports.end ()) {\r
-            std::cerr << "WavesAudioBackend::_unregister_system_midi_ports (): Failed to find port [" << (*it)->name () << "]!"  << std::endl;\r
-        }\r
-        else\r
-            _ports.erase (port_iterator);\r
-        delete *it;\r
-    }\r
-    _physical_midi_inputs.clear ();\r
-    _physical_midi_outputs.clear ();\r
-}\r
-\r
-\r
-int\r
-WavesAudioBackend::_register_system_midi_ports ()\r
-{\r
-    // COMMENTED DBG LOGS */ std::cout  << "WavesAudioBackend::_register_system_midi_ports ()" << std::endl;\r
-\r
-    LatencyRange lr = {0,0};\r
-    lr.min = lr.max = _buffer_size;\r
-\r
-    for (size_t i = 0; i<_ports.size ();)    {\r
-        WavesMidiPort* midi_port = dynamic_cast<WavesMidiPort*> (_ports[i]);\r
-        if (!midi_port || !midi_port->is_physical () || !midi_port->is_terminal ()) {\r
-            ++i;\r
-            continue;\r
-        }\r
-\r
-        if ((midi_port->is_input () && !midi_port->midi_device ()->is_output ()) ||\r
-            (midi_port->is_output () && !midi_port->midi_device ()->is_input ())) {\r
-            disconnect_all (midi_port);\r
-            unregister_port (midi_port);\r
-            continue; // to be here for further additions in the end of this loop\r
-        }\r
-\r
-        ++i;\r
-    }\r
-\r
-    const std::vector<WavesMidiDevice *>&  devices = _midi_device_manager.devices ();\r
-\r
-    for (std::vector<WavesMidiDevice*>::const_iterator it = devices.begin (); it != devices.end (); ++it) {\r
-        if ((*it)->is_input ()) {\r
-            std::string port_name = "system_midi:" + (*it)->name () + " capture";\r
-            WavesDataPort* port = _find_port (port_name);\r
-            WavesMidiPort* midi_port = dynamic_cast<WavesMidiPort*> (port);\r
-            if (midi_port && (midi_port->type () != DataType::MIDI || \r
-                midi_port->midi_device () != *it || \r
-                !midi_port->is_output () || \r
-                !midi_port->is_physical () ||\r
-                !midi_port->is_terminal ())) {\r
-                std::cerr << "WavesAudioBackend::_register_system_midi_ports (): the port [" << midi_port->name () << "] is inconsystently constructed!" << std::endl;\r
-                disconnect_all (midi_port);\r
-                unregister_port (midi_port);\r
-                port = NULL;\r
-            }\r
-\r
-            if (port == NULL) {\r
-                port = _register_port ( port_name, DataType::MIDI , static_cast<ARDOUR::PortFlags> (IsOutput | IsPhysical | IsTerminal));\r
-                if (port == NULL) {\r
-                    return -1;\r
-                }\r
-                ((WavesMidiPort*)port)->set_midi_device (*it);\r
-            }\r
-            port->set_latency_range (lr, false); \r
-        }\r
-\r
-        if ((*it)->is_output ()) {\r
-            std::string port_name = "system_midi:" + (*it)->name () + " playback";\r
-            WavesDataPort* port = _find_port (port_name);\r
-            WavesMidiPort* midi_port = dynamic_cast<WavesMidiPort*> (port);\r
-            if (midi_port && (midi_port->type () != DataType::MIDI || \r
-                midi_port->midi_device () != *it || \r
-                !midi_port->is_input () || \r
-                !midi_port->is_physical () ||\r
-                !midi_port->is_terminal ())) {\r
-                std::cerr << "WavesAudioBackend::_register_system_midi_ports (): the port [" << midi_port->name () << "] is inconsystently constructed!" << std::endl;\r
-                disconnect_all (midi_port);\r
-                unregister_port (midi_port);\r
-            }\r
-\r
-            if (port == NULL) {\r
-                port = _register_port (port_name,\r
-                                       DataType::MIDI,\r
-                                       static_cast<ARDOUR::PortFlags> (IsInput | IsPhysical | IsTerminal));\r
-                if (port == NULL) {\r
-                    return -1;\r
-                }\r
-            }\r
-\r
-            ((WavesMidiPort*)port)->set_midi_device ((*it));\r
-            port->set_latency_range (lr, true);\r
-        }\r
-    }\r
-    \r
-    return 0;\r
-}\r
-\r
-\r
-int\r
-WavesAudioBackend::_read_midi_data_from_devices ()\r
-{\r
-    // COMMENTED FREQUENT DBG LOGS */ std::cout  << "WavesAudioBackend::_read_midi_data_from_devices ():" << std::endl;\r
-    if (!_midi_device_manager.is_streaming ())\r
-        return 0;\r
-    \r
-    _midi_device_manager.do_read ();\r
-\r
-    for (std::vector<WavesMidiPort*>::iterator it = _physical_midi_inputs.begin (); it != _physical_midi_inputs.end (); ++it) {\r
-        WavesMidiDevice* midi_device = (*it)->midi_device ();\r
-        \r
-        WavesMidiBuffer& waves_midi_buffer = (*it)->buffer ();\r
-        waves_midi_buffer.clear ();\r
-        \r
-        while (WavesMidiEvent *waves_midi_event = midi_device->dequeue_input_waves_midi_event ()) {\r
-            int32_t timestamp_st = _buffer_size - (_sample_time_at_cycle_start - waves_midi_event->timestamp ());\r
-            \r
-            if (timestamp_st < 0) {\r
-                timestamp_st = 0;\r
-            }\r
-            else if (timestamp_st >= (int32_t)_buffer_size) {\r
-                timestamp_st = _buffer_size - 1;\r
-            }\r
-            waves_midi_event->set_timestamp (timestamp_st);\r
-            waves_midi_buffer.push_back (waves_midi_event);\r
-        }\r
-    }\r
-    return 0;\r
-}\r
-\r
-\r
-int\r
-WavesAudioBackend::_write_midi_data_to_devices (pframes_t nframes)\r
-{\r
-    if (!_midi_device_manager.is_streaming ())\r
-        return 0;\r
-    \r
-    for (std::vector<WavesMidiPort*>::iterator it = _physical_midi_outputs.begin (); it != _physical_midi_outputs.end (); ++it) {\r
-        WavesMidiDevice* midi_device = (*it)->midi_device (); \r
-        WavesMidiBuffer &waves_midi_buffer = * (WavesMidiBuffer*) (*it)->get_buffer (nframes);\r
-\r
-        for (WavesMidiBufferIterator it = waves_midi_buffer.begin (); it != waves_midi_buffer.end ();) {\r
-             WavesMidiEvent* waves_midi_event = *it;\r
-            \r
-            waves_midi_buffer.erase (it);\r
-            \r
-            waves_midi_event->set_timestamp (_sample_time_at_cycle_start + waves_midi_event->timestamp () + nframes);\r
-            midi_device->enqueue_output_waves_midi_event (waves_midi_event);\r
-       }\r
-    }\r
-    _midi_device_manager.do_write ();\r
-    return 0;\r
-}\r
+/*
+    Copyright (C) 2014 Waves Audio Ltd.
+
+    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.
+
+*/
+#include <boost/assign/list_of.hpp>
+
+#include "waves_audiobackend.h"
+#include "waves_midiport.h"
+#include "waves_midi_event.h"
+#include "waves_midi_buffer.h"
+
+using namespace ARDOUR;
+
+#ifdef __APPLE__
+
+const std::vector<std::string> WavesAudioBackend::__available_midi_options = boost::assign::list_of ("CoreMIDI") ("None");
+
+#elif PLATFORM_WINDOWS
+
+const std::vector<std::string> WavesAudioBackend::__available_midi_options = boost::assign::list_of ("System MIDI (MME)") ("None");
+
+#endif
+
+
+std::vector<std::string> 
+WavesAudioBackend::enumerate_midi_options () const
+{
+    // COMMENTED DBG LOGS */ std::cout << "WavesAudioBackend::enumerate_midi_options ()" << std::endl;
+    return __available_midi_options;
+}
+
+
+int 
+WavesAudioBackend::set_midi_option (const std::string& option)
+{
+    // COMMENTED DBG LOGS */ std::cout << "WavesAudioBackend::set_midi_option ( " << option << " )" << std::endl;
+    if (option == __available_midi_options[1]) {
+        _use_midi = false;
+        // COMMENTED DBG LOGS */ std::cout << "\tNO MIDI system used)" << std::endl;
+    }
+    else if (option == __available_midi_options[0]) {
+        _use_midi = true;
+        // COMMENTED DBG LOGS */ std::cout << "\tNO MIDI system used)" << std::endl;
+    }
+    else {
+        std::cerr << "WavesAudioBackend::set_midi_option (): Invalid MIDI option!" << std::endl;
+        return -1;
+    }
+
+    return 0;
+}
+
+
+std::string
+WavesAudioBackend::midi_option () const
+{
+    // COMMENTED DBG LOGS */ std::cout << "WavesAudioBackend::midi_option ():" << std::endl;
+    return * (__available_midi_options.begin () + (_use_midi?0:1));
+}
+
+
+int
+WavesAudioBackend::midi_event_get (pframes_t& timestamp, size_t& size, uint8_t** buffer, void* port_buffer, uint32_t event_index)
+{
+    // COMMENTED FREQUENT DBG LOGS */ std::cout << "WavesAudioBackend::midi_event_get ():" << std::endl;
+
+    if (buffer == NULL) {
+        std::cerr << "WavesAudioBackend::midi_event_get () : NULL in the 'buffer' argument!\n";
+        return -1;
+    }
+
+    if (port_buffer == NULL) {
+        std::cerr << "WavesAudioBackend::midi_event_get () : NULL in the 'port_buffer' argument!\n";
+        return -1;
+    }
+
+    WavesMidiBuffer& source = * (WavesMidiBuffer*)port_buffer;
+
+    if (event_index >= source.size ()) {
+        std::cerr << "WavesAudioBackend::midi_event_get () : 'event_index' is out of the number of events stored in 'port_buffer'!\n";
+        return -1;
+    }
+
+    WavesMidiEvent* waves_midi_event = source[event_index];
+
+    timestamp = waves_midi_event->timestamp ();
+    size = waves_midi_event->size ();
+    *buffer = waves_midi_event->data ();
+
+    return 0;
+}
+
+
+int
+WavesAudioBackend::midi_event_put (void* port_buffer, pframes_t timestamp, const uint8_t* buffer, size_t size)
+{
+    // COMMENTED FREQUENT DBG LOGS */ std::cout << "WavesAudioBackend::midi_event_put ():" << std::endl;
+    if (buffer == NULL) {
+        std::cerr << "WavesAudioBackend::midi_event_put () : NULL in the 'buffer' argument!\n";
+        return -1;
+    }
+
+    if (port_buffer == NULL) {
+        std::cerr << "WavesAudioBackend::midi_event_put () : NULL in the 'port_buffer' argument!\n";
+        return -1;
+    }
+
+    WavesMidiBuffer& target = * (WavesMidiBuffer*)port_buffer;
+    // COMMENTED FREQUENT DBG LOGS */ std::cout << "\t [" << target.name () << "]"<< std::endl;
+
+    if (target.size () && (pframes_t)target.back ()->timestamp () > timestamp) {
+        std::cerr << "WavesAudioBackend::midi_event_put (): The MIDI Event to put is a bit late!" << std::endl;
+        std::cerr << "\tprev timestamp is " << (pframes_t)target.back ()->timestamp () << " as the current one is " << timestamp << std::endl;
+        return -1;
+    }
+
+    target.push_back (new WavesMidiEvent (timestamp, buffer, size));
+    return 0;
+}
+
+
+uint32_t
+WavesAudioBackend::get_midi_event_count (void* port_buffer)
+{
+    // COMMENTED FREQUENT DBG LOGS */ std::cout << "WavesAudioBackend::get_midi_event_count (): " << std::endl;
+    
+    if (port_buffer == NULL) {
+        std::cerr << "WavesAudioBackend::get_midi_event_count () : NULL in the 'port_buffer' argument!\n";
+        return -1;
+    }
+
+    // COMMENTED FREQUENT DBG LOGS */ std::cout << "\tcount = " << (* (WavesMidiBuffer*)port_buffer).size () << std::endl;
+
+    return (* (WavesMidiBuffer*)port_buffer).size ();
+}
+
+
+void
+WavesAudioBackend::midi_clear (void* port_buffer)
+{
+    // COMMENTED FREQUENT DBG LOGS */ std::cout << "WavesAudioBackend::midi_clear (): " << std::endl;
+    if (port_buffer == NULL) {
+        std::cerr << "WavesAudioBackend::midi_clear () : NULL in the 'port_buffer' argument!\n";
+        return;
+    }
+
+    (* (WavesMidiBuffer*)port_buffer).clear ();
+}
+
+
+void
+WavesAudioBackend::_changed_midi_devices ()
+{
+    if (_midi_device_manager.stream (false)) {
+        std::cerr << "WavesAudioBackend::_changed_midi_devices (): _midi_device_manager.stream (false) failed!" << std::endl;
+        return;
+    }
+
+       _unregister_system_midi_ports ();
+    _midi_device_manager.stop ();
+
+    if (_midi_device_manager.start () != 0) {
+        std::cerr << "WavesAudioBackend::_changed_midi_devices (): _midi_device_manager.start () failed!" << std::endl;
+        return;
+    }
+
+    if (_register_system_midi_ports () != 0) {
+        std::cerr << "WavesAudioBackend::_changed_midi_devices (): _register_system_midi_ports () failed!" << std::endl;
+        return;
+    }
+    
+    manager.registration_callback ();
+
+    if (_midi_device_manager.stream (true)) {
+        std::cerr << "WavesAudioBackend::_changed_midi_devices (): _midi_device_manager.stream (true) failed!" << std::endl;
+        return;
+    }
+}
+
+
+void
+WavesAudioBackend::_unregister_system_midi_ports ()
+{
+    // COMMENTED DBG LOGS */ std::cout  << "WavesAudioBackend::_unregister_system_midi_ports ()" << std::endl;
+    std::vector<WavesMidiPort*> physical_midi_ports = _physical_midi_inputs;
+    physical_midi_ports.insert (physical_midi_ports.begin (), _physical_midi_outputs.begin (), _physical_midi_outputs.end ());
+
+    for (std::vector<WavesMidiPort*>::const_iterator it = physical_midi_ports.begin (); it != physical_midi_ports.end (); ++it) {
+        std::vector<WavesDataPort*>::iterator port_iterator = std::find (_ports.begin (), _ports.end (), *it);
+        if (port_iterator == _ports.end ()) {
+            std::cerr << "WavesAudioBackend::_unregister_system_midi_ports (): Failed to find port [" << (*it)->name () << "]!"  << std::endl;
+        }
+        else
+            _ports.erase (port_iterator);
+        delete *it;
+    }
+    _physical_midi_inputs.clear ();
+    _physical_midi_outputs.clear ();
+}
+
+
+int
+WavesAudioBackend::_register_system_midi_ports ()
+{
+    // COMMENTED DBG LOGS */ std::cout  << "WavesAudioBackend::_register_system_midi_ports ()" << std::endl;
+
+    LatencyRange lr = {0,0};
+    lr.min = lr.max = _buffer_size;
+
+    for (size_t i = 0; i<_ports.size ();)    {
+        WavesMidiPort* midi_port = dynamic_cast<WavesMidiPort*> (_ports[i]);
+        if (!midi_port || !midi_port->is_physical () || !midi_port->is_terminal ()) {
+            ++i;
+            continue;
+        }
+
+        if ((midi_port->is_input () && !midi_port->midi_device ()->is_output ()) ||
+            (midi_port->is_output () && !midi_port->midi_device ()->is_input ())) {
+            disconnect_all (midi_port);
+            unregister_port (midi_port);
+            continue; // to be here for further additions in the end of this loop
+        }
+
+        ++i;
+    }
+
+    const std::vector<WavesMidiDevice *>&  devices = _midi_device_manager.devices ();
+
+    for (std::vector<WavesMidiDevice*>::const_iterator it = devices.begin (); it != devices.end (); ++it) {
+        if ((*it)->is_input ()) {
+            std::string port_name = "system_midi:" + (*it)->name () + " capture";
+            WavesDataPort* port = _find_port (port_name);
+            WavesMidiPort* midi_port = dynamic_cast<WavesMidiPort*> (port);
+            if (midi_port && (midi_port->type () != DataType::MIDI || 
+                midi_port->midi_device () != *it || 
+                !midi_port->is_output () || 
+                !midi_port->is_physical () ||
+                !midi_port->is_terminal ())) {
+                std::cerr << "WavesAudioBackend::_register_system_midi_ports (): the port [" << midi_port->name () << "] is inconsystently constructed!" << std::endl;
+                disconnect_all (midi_port);
+                unregister_port (midi_port);
+                port = NULL;
+            }
+
+            if (port == NULL) {
+                port = _register_port ( port_name, DataType::MIDI , static_cast<ARDOUR::PortFlags> (IsOutput | IsPhysical | IsTerminal));
+                if (port == NULL) {
+                    return -1;
+                }
+                ((WavesMidiPort*)port)->set_midi_device (*it);
+            }
+            port->set_latency_range (lr, false); 
+        }
+
+        if ((*it)->is_output ()) {
+            std::string port_name = "system_midi:" + (*it)->name () + " playback";
+            WavesDataPort* port = _find_port (port_name);
+            WavesMidiPort* midi_port = dynamic_cast<WavesMidiPort*> (port);
+            if (midi_port && (midi_port->type () != DataType::MIDI || 
+                midi_port->midi_device () != *it || 
+                !midi_port->is_input () || 
+                !midi_port->is_physical () ||
+                !midi_port->is_terminal ())) {
+                std::cerr << "WavesAudioBackend::_register_system_midi_ports (): the port [" << midi_port->name () << "] is inconsystently constructed!" << std::endl;
+                disconnect_all (midi_port);
+                unregister_port (midi_port);
+            }
+
+            if (port == NULL) {
+                port = _register_port (port_name,
+                                       DataType::MIDI,
+                                       static_cast<ARDOUR::PortFlags> (IsInput | IsPhysical | IsTerminal));
+                if (port == NULL) {
+                    return -1;
+                }
+            }
+
+            ((WavesMidiPort*)port)->set_midi_device ((*it));
+            port->set_latency_range (lr, true);
+        }
+    }
+    
+    return 0;
+}
+
+
+int
+WavesAudioBackend::_read_midi_data_from_devices ()
+{
+    // COMMENTED FREQUENT DBG LOGS */ std::cout  << "WavesAudioBackend::_read_midi_data_from_devices ():" << std::endl;
+    if (!_midi_device_manager.is_streaming ())
+        return 0;
+    
+    _midi_device_manager.do_read ();
+
+    for (std::vector<WavesMidiPort*>::iterator it = _physical_midi_inputs.begin (); it != _physical_midi_inputs.end (); ++it) {
+        WavesMidiDevice* midi_device = (*it)->midi_device ();
+        
+        WavesMidiBuffer& waves_midi_buffer = (*it)->buffer ();
+        waves_midi_buffer.clear ();
+        
+        while (WavesMidiEvent *waves_midi_event = midi_device->dequeue_input_waves_midi_event ()) {
+            int32_t timestamp_st = _buffer_size - (_sample_time_at_cycle_start - waves_midi_event->timestamp ());
+            
+            if (timestamp_st < 0) {
+                timestamp_st = 0;
+            } else if (timestamp_st >= (int32_t)_buffer_size) {
+                timestamp_st = _buffer_size - 1;
+            }
+            waves_midi_event->set_timestamp (timestamp_st);
+            waves_midi_buffer.push_back (waves_midi_event);
+        }
+    }
+    return 0;
+}
+
+
+int
+WavesAudioBackend::_write_midi_data_to_devices (pframes_t nframes)
+{
+    if (!_midi_device_manager.is_streaming ())
+        return 0;
+    
+    for (std::vector<WavesMidiPort*>::iterator it = _physical_midi_outputs.begin (); it != _physical_midi_outputs.end (); ++it) {
+        WavesMidiDevice* midi_device = (*it)->midi_device (); 
+        WavesMidiBuffer &waves_midi_buffer = * (WavesMidiBuffer*) (*it)->get_buffer (nframes);
+
+        for (WavesMidiBufferIterator it = waves_midi_buffer.begin (); it != waves_midi_buffer.end ();) {
+             WavesMidiEvent* waves_midi_event = *it;
+            
+            waves_midi_buffer.erase (it);
+            
+            waves_midi_event->set_timestamp (_sample_time_at_cycle_start + waves_midi_event->timestamp () + nframes);
+            midi_device->enqueue_output_waves_midi_event (waves_midi_event);
+       }
+    }
+    _midi_device_manager.do_write ();
+    return 0;
+}