/* * Copyright (C) 2013-2019 Paul Davis * Copyright (C) 2015-2017 Robin Gareus * * 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., * 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA. */ #ifndef __libardour_port_manager_h__ #define __libardour_port_manager_h__ #include #include #include #include #include #include #include "pbd/rcu.h" #include "pbd/ringbuffer.h" #include "ardour/chan_count.h" #include "ardour/midiport_manager.h" #include "ardour/port.h" namespace ARDOUR { class PortEngine; class AudioBackend; class Session; class LIBARDOUR_API PortManager { public: typedef std::map > Ports; typedef std::list > PortList; PortManager (); virtual ~PortManager () {} PortEngine& port_engine (); uint32_t port_name_size () const; std::string my_name () const; /* Port registration */ boost::shared_ptr register_input_port (DataType, const std::string& portname, bool async = false, PortFlags extra_flags = PortFlags (0)); boost::shared_ptr register_output_port (DataType, const std::string& portname, bool async = false, PortFlags extra_flags = PortFlags (0)); int unregister_port (boost::shared_ptr); /* Port connectivity */ int connect (const std::string& source, const std::string& destination); int disconnect (const std::string& source, const std::string& destination); int disconnect (boost::shared_ptr); int disconnect (std::string const&); int reestablish_ports (); int reconnect_ports (); bool connected (const std::string&); bool physically_connected (const std::string&); int get_connections (const std::string&, std::vector&); /* Naming */ boost::shared_ptr get_port_by_name (const std::string&); void port_renamed (const std::string&, const std::string&); std::string make_port_name_relative (const std::string& name) const; std::string make_port_name_non_relative (const std::string& name) const; std::string get_pretty_name_by_name (const std::string& portname) const; std::string short_port_name_from_port_name (std::string const& full_name) const; bool port_is_mine (const std::string& fullname) const; static bool port_is_control_only (std::string const&); /* other Port management */ bool port_is_physical (const std::string&) const; void get_physical_outputs (DataType type, std::vector&, MidiPortFlags include = MidiPortFlags (0), MidiPortFlags exclude = MidiPortFlags (0)); void get_physical_inputs (DataType type, std::vector&, MidiPortFlags include = MidiPortFlags (0), MidiPortFlags exclude = MidiPortFlags (0)); ChanCount n_physical_outputs () const; ChanCount n_physical_inputs () const; int get_ports (const std::string& port_name_pattern, DataType type, PortFlags flags, std::vector&); int get_ports (DataType, PortList&); void set_port_pretty_name (std::string const&, std::string const&); void remove_all_ports (); void clear_pending_port_deletions (); virtual void add_pending_port_deletion (Port*) = 0; PBD::RingBuffer& port_deletions_pending () { return _port_deletions_pending; } bool check_for_ambiguous_latency (bool log = false) const; /* per-Port monitoring */ bool can_request_input_monitoring () const; void request_input_monitoring (const std::string&, bool) const; void ensure_input_monitoring (const std::string&, bool) const; class PortRegistrationFailure : public std::exception { public: PortRegistrationFailure (std::string const& why = "") : reason (why) {} ~PortRegistrationFailure () throw () {} const char* what () const throw () { return reason.c_str (); } private: std::string reason; }; /* the port engine will invoke these callbacks when the time is right */ void registration_callback (); int graph_order_callback (); void connect_callback (const std::string&, const std::string&, bool connection); bool port_remove_in_progress () const { return _port_remove_in_progress; } struct MidiPortInformation { std::string backend; std::string pretty_name; bool input; MidiPortFlags properties; bool exists; MidiPortInformation (std::string const& b, std::string const& pretty, bool input, MidiPortFlags flags, bool xists) : backend (b) , pretty_name (pretty) , input (input) , properties (flags) , exists (xists) {} MidiPortInformation () : properties (MidiPortFlags (0)) , exists (false) {} }; void fill_midi_port_info (); MidiPortInformation midi_port_information (std::string const&); void get_known_midi_ports (std::vector&); void get_midi_selection_ports (std::vector&); void add_midi_port_flags (std::string const&, MidiPortFlags); void remove_midi_port_flags (std::string const&, MidiPortFlags); /** Emitted if the list of ports to be used for MIDI selection tracking changes */ PBD::Signal0 MidiSelectionPortsChanged; /** Emitted if anything other than the selection property for a MIDI port changes */ PBD::Signal0 MidiPortInfoChanged; /** Emitted if the backend notifies us of a graph order event */ PBD::Signal0 GraphReordered; /** Emitted if a Port is registered or unregistered */ PBD::Signal0 PortRegisteredOrUnregistered; /** Emitted if a Port is connected or disconnected. * The Port parameters are the ports being connected / disconnected, or 0 if they are not known to Ardour. * The std::string parameters are the (long) port names. * The bool parameter is true if ports were connected, or false for disconnected. */ PBD::Signal5, std::string, boost::weak_ptr, std::string, bool> PortConnectedOrDisconnected; protected: boost::shared_ptr _backend; SerializedRCUManager ports; bool _port_remove_in_progress; PBD::RingBuffer _port_deletions_pending; boost::shared_ptr register_port (DataType type, const std::string& portname, bool input, bool async = false, PortFlags extra_flags = PortFlags (0)); void port_registration_failure (const std::string& portname); /** List of ports to be used between \ref cycle_start() and \ref cycle_end() */ boost::shared_ptr _cycle_ports; void silence (pframes_t nframes, Session* s = 0); void silence_outputs (pframes_t nframes); void check_monitoring (); /** Signal the start of an audio cycle. * This MUST be called before any reading/writing for this cycle. * Realtime safe. */ void cycle_start (pframes_t nframes, Session* s = 0); /** Signal the end of an audio cycle. * This signifies that the cycle began with @ref cycle_start has ended. * This MUST be called at the end of each cycle. * Realtime safe. */ void cycle_end (pframes_t nframes, Session* s = 0); void cycle_end_fade_out (gain_t, gain_t, pframes_t, Session* s = 0); typedef std::map MidiPortInfo; mutable Glib::Threads::Mutex midi_port_info_mutex; MidiPortInfo midi_port_info; static std::string midi_port_info_file (); bool midi_info_dirty; void save_midi_port_info (); void load_midi_port_info (); void fill_midi_port_info_locked (); void filter_midi_ports (std::vector&, MidiPortFlags, MidiPortFlags); void set_port_buffer_sizes (pframes_t); }; } // namespace ARDOUR #endif /* __libardour_port_manager_h__ */