// Copyright 2007-2020 David Robillard // SPDX-License-Identifier: GPL-3.0-or-later #include "AudioDriver.hpp" #include "ClientID.hpp" #include "ClientInfo.hpp" #include "ClientType.hpp" #include "Driver.hpp" #include "Event.hpp" #include "ILog.hpp" #include "PortID.hpp" #include "PortInfo.hpp" #include "PortNames.hpp" #include "PortType.hpp" #include "SignalDirection.hpp" #include "jackey.h" #include "make_jack_driver.hpp" #include "patchage_config.h" #include "warnings.hpp" #if USE_JACK_METADATA # include #endif PATCHAGE_DISABLE_FMT_WARNINGS #include PATCHAGE_RESTORE_WARNINGS #include #include #include #include #include #include #include #include #include #include #include #include #include namespace patchage { namespace { /// Driver for JACK audio and midi ports that uses libjack class JackLibDriver : public AudioDriver { public: explicit JackLibDriver(ILog& log, EventSink emit_event); JackLibDriver(const JackLibDriver&) = delete; JackLibDriver& operator=(const JackLibDriver&) = delete; JackLibDriver(JackLibDriver&&) = delete; JackLibDriver& operator=(JackLibDriver&&) = delete; ~JackLibDriver() override; // Driver interface void attach(bool launch_daemon) override; void detach() override; bool is_attached() const override; void refresh(const EventSink& sink) override; bool connect(const PortID& tail_id, const PortID& head_id) override; bool disconnect(const PortID& tail_id, const PortID& head_id) override; // AudioDriver interface uint32_t xruns() override; void reset_xruns() override; uint32_t buffer_size() override; bool set_buffer_size(uint32_t frames) override; uint32_t sample_rate() override; private: static ClientInfo get_client_info(const char* name); PortInfo get_port_info(const jack_port_t* port); static void on_client(const char* name, int registered, void* driver); static void on_port(jack_port_id_t port_id, int registered, void* driver); static void on_connection(jack_port_id_t src, jack_port_id_t dst, int connect, void* driver); static int on_xrun(void* driver); static void on_shutdown(void* driver); ILog& _log; std::mutex _shutdown_mutex; jack_client_t* _client = nullptr; jack_nframes_t _buffer_size = 0U; uint32_t _xruns = 0U; bool _is_activated = false; }; JackLibDriver::JackLibDriver(ILog& log, EventSink emit_event) : AudioDriver{std::move(emit_event)} , _log{log} {} JackLibDriver::~JackLibDriver() { detach(); } void JackLibDriver::attach(const bool launch_daemon) { if (_client) { return; // Already connected } const jack_options_t options = (!launch_daemon) ? JackNoStartServer : JackNullOption; if (!(_client = jack_client_open("Patchage", options, nullptr))) { _log.error("[JACK] Unable to create client"); _is_activated = false; return; } jack_on_shutdown(_client, on_shutdown, this); jack_set_client_registration_callback(_client, on_client, this); jack_set_port_registration_callback(_client, on_port, this); jack_set_port_connect_callback(_client, on_connection, this); jack_set_xrun_callback(_client, on_xrun, this); if (jack_activate(_client)) { _log.error("[JACK] Client activation failed"); _is_activated = false; _buffer_size = 0; return; } _is_activated = true; _buffer_size = jack_get_buffer_size(_client); _emit_event(event::DriverAttached{ClientType::jack}); } void JackLibDriver::detach() { const std::lock_guard lock{_shutdown_mutex}; if (_client) { jack_deactivate(_client); jack_client_close(_client); _client = nullptr; } _is_activated = false; _emit_event(event::DriverDetached{ClientType::jack}); } bool JackLibDriver::is_attached() const { return _client != nullptr; } std::string get_property(const jack_uuid_t subject, const char* const key) { std::string result; #if USE_JACK_METADATA char* value = nullptr; char* datatype = nullptr; if (!jack_get_property(subject, key, &value, &datatype)) { result = value; } jack_free(datatype); jack_free(value); #else (void)subject; (void)key; #endif return result; } ClientInfo JackLibDriver::get_client_info(const char* const name) { return {name}; // TODO: Pretty name? } PortInfo JackLibDriver::get_port_info(const jack_port_t* const port) { const auto uuid = jack_port_uuid(port); const auto flags = static_cast(jack_port_flags(port)); const std::string name = jack_port_name(port); auto label = PortNames{name}.port(); // Get pretty name to use as a label, if present #if USE_JACK_METADATA const auto pretty_name = get_property(uuid, JACK_METADATA_PRETTY_NAME); if (!pretty_name.empty()) { label = pretty_name; } #endif // Determine detailed type, using metadata for fancy types if possible const char* const type_str = jack_port_type(port); PortType type = PortType::jack_audio; if (!strcmp(type_str, JACK_DEFAULT_AUDIO_TYPE)) { if (get_property(uuid, JACKEY_SIGNAL_TYPE) == "CV") { type = PortType::jack_cv; } } else if (!strcmp(type_str, JACK_DEFAULT_MIDI_TYPE)) { type = PortType::jack_midi; if (get_property(uuid, JACKEY_EVENT_TYPES) == "OSC") { type = PortType::jack_osc; } } else { _log.warning( fmt::format(R"([JACK] Port "{}" has unknown type "{}")", name, type_str)); } // Get direction from port flags const SignalDirection direction = ((flags & JackPortIsInput) ? SignalDirection::input : SignalDirection::output); // Get port order from metadata if possible std::optional order; const std::string order_str = get_property(uuid, JACKEY_ORDER); if (!order_str.empty()) { order = std::stoi(order_str); } return {label, type, direction, order, static_cast(flags & JackPortIsTerminal)}; } void JackLibDriver::refresh(const EventSink& sink) { const std::lock_guard lock{_shutdown_mutex}; if (!_client) { return; } // Get all existing ports const char** const ports = jack_get_ports(_client, nullptr, nullptr, 0); if (!ports) { return; } // Get all client names (to only send a creation event once for each) std::unordered_set client_names; for (auto i = 0U; ports[i]; ++i) { client_names.insert(PortID::jack(ports[i]).client().jack_name()); } // Emit all clients for (const auto& client_name : client_names) { sink({event::ClientCreated{ClientID::jack(client_name), get_client_info(client_name.c_str())}}); } // Emit all ports for (auto i = 0U; ports[i]; ++i) { const jack_port_t* const port = jack_port_by_name(_client, ports[i]); sink({event::PortCreated{PortID::jack(ports[i]), get_port_info(port)}}); } // Get all connections (again to only create them once) std::set> connections; for (auto i = 0U; ports[i]; ++i) { const jack_port_t* const port = jack_port_by_name(_client, ports[i]); const char** const peers = jack_port_get_all_connections(_client, port); if (peers) { const auto flags = static_cast(jack_port_flags(port)); if (flags & JackPortIsInput) { for (auto j = 0U; peers[j]; ++j) { connections.emplace(peers[j], ports[i]); } } else { for (auto j = 0U; peers[j]; ++j) { connections.emplace(ports[i], peers[j]); } } jack_free(peers); } } // Emit all connections for (const auto& connection : connections) { sink({event::PortsConnected{PortID::jack(connection.first), PortID::jack(connection.second)}}); } jack_free(ports); } bool JackLibDriver::connect(const PortID& tail_id, const PortID& head_id) { if (!_client) { return false; } const auto& tail_name = tail_id.jack_name(); const auto& head_name = head_id.jack_name(); const int result = jack_connect(_client, tail_name.c_str(), head_name.c_str()); if (result) { _log.error( fmt::format("[JACK] Failed to connect {} => {}", tail_name, head_name)); return false; } return true; } bool JackLibDriver::disconnect(const PortID& tail_id, const PortID& head_id) { if (!_client) { return false; } const auto& tail_name = tail_id.jack_name(); const auto& head_name = head_id.jack_name(); const int result = jack_disconnect(_client, tail_name.c_str(), head_name.c_str()); if (result) { _log.error(fmt::format( "[JACK] Failed to disconnect {} => {}", tail_name, head_name)); return false; } return true; } uint32_t JackLibDriver::xruns() { return _xruns; } void JackLibDriver::reset_xruns() { _xruns = 0; } uint32_t JackLibDriver::buffer_size() { return _is_activated ? _buffer_size : jack_get_buffer_size(_client); } bool JackLibDriver::set_buffer_size(const uint32_t frames) { if (!_client) { _buffer_size = frames; return true; } if (buffer_size() == frames) { return true; } if (jack_set_buffer_size(_client, frames)) { _log.error("[JACK] Unable to set buffer size"); return false; } _buffer_size = frames; return true; } uint32_t JackLibDriver::sample_rate() { return jack_get_sample_rate(_client); } void JackLibDriver::on_client(const char* const name, const int registered, void* const driver) { auto* const me = static_cast(driver); if (registered) { me->_emit_event(event::ClientCreated{ClientID::jack(name), {name}}); } else { me->_emit_event(event::ClientDestroyed{ClientID::jack(name)}); } } void JackLibDriver::on_port(const jack_port_id_t port_id, const int registered, void* const driver) { auto* const me = static_cast(driver); jack_port_t* const port = jack_port_by_id(me->_client, port_id); const char* const name = jack_port_name(port); const auto id = PortID::jack(name); if (registered) { me->_emit_event(event::PortCreated{id, me->get_port_info(port)}); } else { me->_emit_event(event::PortDestroyed{id}); } } void JackLibDriver::on_connection(const jack_port_id_t src, const jack_port_id_t dst, const int connect, void* const driver) { auto* const me = static_cast(driver); jack_port_t* const src_port = jack_port_by_id(me->_client, src); jack_port_t* const dst_port = jack_port_by_id(me->_client, dst); const char* const src_name = jack_port_name(src_port); const char* const dst_name = jack_port_name(dst_port); if (connect) { me->_emit_event( event::PortsConnected{PortID::jack(src_name), PortID::jack(dst_name)}); } else { me->_emit_event( event::PortsDisconnected{PortID::jack(src_name), PortID::jack(dst_name)}); } } int JackLibDriver::on_xrun(void* const driver) { auto* const me = static_cast(driver); ++me->_xruns; return 0; } void JackLibDriver::on_shutdown(void* const driver) { auto* const me = static_cast(driver); /* Note that the JACK documentation lies about this situation. It says the client must not call jack_client_close() here... except that is exactly what libjack does if a shutdown callback isn't registered. Despite that, doing so here hangs forever. Handling it "properly" like a signal handler and calling jack_client_close() in another thread also hangs. So, since JACK is a hot mess and it's impossible to gracefully handle this situation, just leak the client. */ const std::lock_guard lock{me->_shutdown_mutex}; me->_client = nullptr; me->_is_activated = false; me->_emit_event(event::DriverDetached{ClientType::jack}); } } // namespace std::unique_ptr make_jack_driver(ILog& log, Driver::EventSink emit_event) { return std::unique_ptr{ new JackLibDriver{log, std::move(emit_event)}}; } } // namespace patchage