| 
							- /*
 -  * Carla JACK API for external applications
 -  * Copyright (C) 2016-2022 Filipe Coelho <falktx@falktx.com>
 -  *
 -  * 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 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.
 -  *
 -  * For a full copy of the GNU General Public License see the doc/GPL.txt file.
 -  */
 - 
 - #include "libjack.hpp"
 - 
 - #include "CarlaStringList.hpp"
 - 
 - CARLA_BACKEND_USE_NAMESPACE
 - 
 - // --------------------------------------------------------------------------------------------------------------------
 - 
 - static const char* allocate_port_name(const char* const prefixOrFullName, const uint num = UINT32_MAX)
 - {
 -     static CarlaStringList portList;
 - 
 -     char portName[STR_MAX];
 -     carla_zeroChars(portName, STR_MAX);
 - 
 -     if (num == UINT32_MAX)
 -         std::strncpy(portName, prefixOrFullName, STR_MAX-1);
 -     else
 -         std::snprintf(portName, STR_MAX-1, "%s%u", prefixOrFullName, num+1);
 - 
 -     if (const char* const storedPortName = portList.containsAndReturnString(portName))
 -         return storedPortName;
 - 
 -     CARLA_SAFE_ASSERT_RETURN(portList.append(portName), nullptr);
 - 
 -     return portList.getLast();
 - }
 - 
 - // --------------------------------------------------------------------------------------------------------------------
 - 
 - CARLA_PLUGIN_EXPORT
 - const char** jack_get_ports(jack_client_t* const client,
 -                             const char* const port_name,
 -                             const char* const port_type,
 -                             const unsigned long flags)
 - {
 -     carla_stdout("%s(%p, %s, %s, 0x%lx)", __FUNCTION__, client, port_name, port_type, flags);
 - 
 -     JackClientState* const jclient = (JackClientState*)client;
 -     CARLA_SAFE_ASSERT_RETURN(jclient != nullptr, nullptr);
 - 
 -     const JackServerState& jserver(jclient->server);
 - 
 -     const uint numIns  = static_cast<uint>(jclient->audioIns.count() +
 -                                            jclient->midiIns.count() +
 -                                            jserver.numAudioIns +
 -                                            jserver.numMidiIns);
 -     const uint numOuts = static_cast<uint>(jclient->audioOuts.count() +
 -                                            jclient->midiOuts.count() +
 -                                            jserver.numAudioOuts +
 -                                            jserver.numMidiOuts);
 - 
 -     const bool wantsAudio = port_type == nullptr || port_type[0] == '\0' || std::strstr(port_type, "audio") != nullptr;
 -     const bool wantsMIDI  = port_type == nullptr || port_type[0] == '\0' || std::strstr(port_type, "midi") != nullptr;
 - 
 -     if (flags == 0x0 || (flags & (JackPortIsInput|JackPortIsOutput)) == (JackPortIsInput|JackPortIsOutput))
 -     {
 -         if (const char** const ret = (const char**)calloc(numIns+numOuts+1, sizeof(const char*)))
 -         {
 -             uint i=0;
 - 
 -             if (wantsAudio)
 -             {
 -                 for (uint j=0; j<jserver.numAudioIns; ++i, ++j)
 -                     ret[i] = allocate_port_name("system:capture_", j);
 -                 for (uint j=0; j<jserver.numAudioOuts; ++i, ++j)
 -                     ret[i] = allocate_port_name("system:playback_", j);
 -             }
 -             if (wantsMIDI)
 -             {
 -                 for (uint j=0; j<jserver.numMidiIns; ++i, ++j)
 -                     ret[i] = allocate_port_name("system:midi_capture_", j);
 -                 for (uint j=0; j<jserver.numMidiOuts; ++i, ++j)
 -                     ret[i] = allocate_port_name("system:midi_playback_", j);
 -             }
 - 
 -             if ((flags & (JackPortIsPhysical|JackPortIsTerminal)) == 0x0)
 -             {
 -                 if (wantsAudio)
 -                 {
 -                     for (LinkedList<JackPortState*>::Itenerator it = jclient->audioIns.begin2(); it.valid(); it.next())
 -                     {
 -                         JackPortState* const jport = it.getValue(nullptr);
 -                         CARLA_SAFE_ASSERT_CONTINUE(jport != nullptr);
 -                         ret[i++] = allocate_port_name(jport->fullname);
 -                     }
 - 
 -                     for (LinkedList<JackPortState*>::Itenerator it = jclient->audioOuts.begin2(); it.valid(); it.next())
 -                     {
 -                         JackPortState* const jport = it.getValue(nullptr);
 -                         CARLA_SAFE_ASSERT_CONTINUE(jport != nullptr);
 -                         ret[i++] = allocate_port_name(jport->fullname);
 -                     }
 -                 }
 - 
 -                 if (wantsMIDI)
 -                 {
 -                     for (LinkedList<JackPortState*>::Itenerator it = jclient->midiIns.begin2(); it.valid(); it.next())
 -                     {
 -                         JackPortState* const jport = it.getValue(nullptr);
 -                         CARLA_SAFE_ASSERT_CONTINUE(jport != nullptr);
 -                         ret[i++] = allocate_port_name(jport->fullname);
 -                     }
 - 
 -                     for (LinkedList<JackPortState*>::Itenerator it = jclient->midiOuts.begin2(); it.valid(); it.next())
 -                     {
 -                         JackPortState* const jport = it.getValue(nullptr);
 -                         CARLA_SAFE_ASSERT_CONTINUE(jport != nullptr);
 -                         ret[i++] = allocate_port_name(jport->fullname);
 -                     }
 -                 }
 -             }
 - 
 -             ret[i] = nullptr;
 - 
 -             return ret;
 -         }
 - 
 -         return nullptr;
 -     }
 - 
 -     if (flags & JackPortIsInput)
 -     {
 -         if (const char** const ret = (const char**)calloc(numIns+1, sizeof(const char*)))
 -         {
 -             uint i=0;
 -             if (wantsAudio)
 -             {
 -                 for (uint j=0; j<jserver.numAudioOuts; ++i, ++j)
 -                     ret[i] = allocate_port_name("system:playback_", j);
 -             }
 -             if (wantsMIDI)
 -             {
 -                 for (uint j=0; j<jserver.numMidiOuts; ++i, ++j)
 -                     ret[i] = allocate_port_name("system:midi_playback_", j);
 -             }
 - 
 -             if ((flags & (JackPortIsPhysical|JackPortIsTerminal)) == 0x0)
 -             {
 -                 if (wantsAudio)
 -                 {
 -                     for (LinkedList<JackPortState*>::Itenerator it = jclient->audioIns.begin2(); it.valid(); it.next())
 -                     {
 -                         JackPortState* const jport = it.getValue(nullptr);
 -                         CARLA_SAFE_ASSERT_CONTINUE(jport != nullptr);
 -                         ret[i++] = allocate_port_name(jport->fullname);
 -                     }
 -                 }
 -                 if (wantsMIDI)
 -                 {
 -                     for (LinkedList<JackPortState*>::Itenerator it = jclient->midiIns.begin2(); it.valid(); it.next())
 -                     {
 -                         JackPortState* const jport = it.getValue(nullptr);
 -                         CARLA_SAFE_ASSERT_CONTINUE(jport != nullptr);
 -                         ret[i++] = allocate_port_name(jport->fullname);
 -                     }
 -                 }
 -             }
 - 
 -             ret[i] = nullptr;
 - 
 -             return ret;
 -         }
 - 
 -         return nullptr;
 -     }
 - 
 -     if (flags & JackPortIsOutput)
 -     {
 -         if (const char** const ret = (const char**)calloc(numOuts+1, sizeof(const char*)))
 -         {
 -             uint i=0;
 - 
 -             if (wantsAudio)
 -             {
 -                 for (uint j=0; j<jserver.numAudioIns; ++i, ++j)
 -                     ret[i] = allocate_port_name("system:capture_", j);
 -             }
 -             if (wantsMIDI)
 -             {
 -                 for (uint j=0; j<jserver.numMidiIns; ++i, ++j)
 -                     ret[i] = allocate_port_name("system:midi_capture_", j);
 -             }
 - 
 -             if ((flags & (JackPortIsPhysical|JackPortIsTerminal)) == 0x0)
 -             {
 -                 if (wantsAudio)
 -                 {
 -                     for (LinkedList<JackPortState*>::Itenerator it = jclient->audioOuts.begin2(); it.valid(); it.next())
 -                     {
 -                         JackPortState* const jport = it.getValue(nullptr);
 -                         CARLA_SAFE_ASSERT_CONTINUE(jport != nullptr);
 -                         ret[i++] = allocate_port_name(jport->fullname);
 -                     }
 -                 }
 -                 if (wantsMIDI)
 -                 {
 -                     for (LinkedList<JackPortState*>::Itenerator it = jclient->midiOuts.begin2(); it.valid(); it.next())
 -                     {
 -                         JackPortState* const jport = it.getValue(nullptr);
 -                         CARLA_SAFE_ASSERT_CONTINUE(jport != nullptr);
 -                         ret[i++] = allocate_port_name(jport->fullname);
 -                     }
 -                 }
 -             }
 - 
 -             ret[i] = nullptr;
 - 
 -             return ret;
 -         }
 - 
 -         return nullptr;
 -     }
 - 
 -     return nullptr;
 - }
 - 
 - CARLA_PLUGIN_EXPORT
 - jack_port_t* jack_port_by_name(jack_client_t* client, const char* name)
 - {
 -     carla_debug("%s(%p, %s)", __FUNCTION__, client, name);
 - 
 -     JackClientState* const jclient = (JackClientState*)client;
 -     CARLA_SAFE_ASSERT_RETURN(jclient != nullptr, nullptr);
 - 
 -     if (std::strncmp(name, "system:", 7) == 0)
 -     {
 -         static std::map<uint, JackPortState*> systemPortIdMapping;
 - 
 -         const JackServerState& jserver(jclient->server);
 -         const int commonFlags = JackPortIsPhysical|JackPortIsTerminal;
 - 
 -         uint rindex, gid;
 -         int flags;
 -         bool isMidi, isConnected;
 - 
 -         const char* const fullname = name;
 -         const char* const portname = name + 7;
 -         name += 7;
 - 
 -         /**/ if (std::strncmp(name, "capture_", 8) == 0)
 -         {
 -             name += 8;
 - 
 -             const int index = std::atoi(name)-1;
 -             CARLA_SAFE_ASSERT_RETURN(index >= 0 && index < jserver.numAudioIns, nullptr);
 - 
 -             rindex = static_cast<uint>(index);
 -             flags  = commonFlags|JackPortIsOutput;
 -             gid    = JackPortState::kPortIdOffsetAudioIn + rindex;
 -             isMidi = false;
 -             isConnected = jserver.numAudioIns > rindex;
 -         }
 -         else if (std::strncmp(name, "playback_", 9) == 0)
 -         {
 -             name += 9;
 - 
 -             const int index = std::atoi(name)-1;
 -             CARLA_SAFE_ASSERT_RETURN(index >= 0 && index < jserver.numAudioOuts, nullptr);
 - 
 -             rindex = static_cast<uint>(jserver.numAudioIns + index);
 -             flags  = commonFlags|JackPortIsInput;
 -             gid    = JackPortState::kPortIdOffsetAudioOut + rindex;
 -             isMidi = false;
 -             isConnected = jserver.numAudioOuts > rindex;
 -         }
 -         else if (std::strncmp(name, "midi_capture_", 13) == 0)
 -         {
 -             name += 13;
 - 
 -             const int index = std::atoi(name)-1;
 -             CARLA_SAFE_ASSERT_RETURN(index >= 0 && index < jserver.numMidiIns, nullptr);
 - 
 -             rindex = static_cast<uint>(index);
 -             flags  = commonFlags|JackPortIsOutput;
 -             gid    = JackPortState::kPortIdOffsetMidiIn + rindex;
 -             isMidi = true;
 -             isConnected = jserver.numMidiIns > rindex;
 -         }
 -         else if (std::strncmp(name, "midi_playback_", 14) == 0)
 -         {
 -             name += 14;
 - 
 -             const int index = std::atoi(name)-1;
 -             CARLA_SAFE_ASSERT_RETURN(index >= 0 && index < jserver.numMidiOuts, nullptr);
 - 
 -             rindex = static_cast<uint>(jserver.numMidiIns + index);
 -             flags  = commonFlags|JackPortIsInput;
 -             gid    = JackPortState::kPortIdOffsetMidiOut + rindex;
 -             isMidi = true;
 -             isConnected = jserver.numMidiOuts > rindex;
 -         }
 -         else
 -         {
 -             carla_stderr2("jack_port_by_name: invalid port short name '%s'", name);
 -             return nullptr;
 -         }
 - 
 -         if (JackPortState* const port = systemPortIdMapping[gid])
 -             return (jack_port_t*)port;
 - 
 -         JackPortState* const port = new JackPortState(fullname,
 -                                                       portname,
 -                                                       rindex, flags, gid,
 -                                                       isMidi, isConnected);
 -         systemPortIdMapping[gid] = port;
 - 
 -         return (jack_port_t*)port;
 -     }
 -     else
 -     {
 -         if (JackPortState* const port = jclient->portNameMapping[name])
 -             return (jack_port_t*)port;
 -     }
 - 
 -     carla_stderr2("jack_port_by_name: invalid port name '%s'", name);
 -     return nullptr;
 - }
 - 
 - CARLA_PLUGIN_EXPORT
 - jack_port_t* jack_port_by_id(jack_client_t* client, jack_port_id_t port_id)
 - {
 -     carla_debug("%s(%p, %u)", __FUNCTION__, client, port_id);
 - 
 -     CARLA_SAFE_ASSERT_UINT_RETURN(port_id >= JackPortState::kPortIdOffsetUser, port_id, nullptr);
 - 
 -     JackClientState* const jclient = (JackClientState*)client;
 -     CARLA_SAFE_ASSERT_RETURN(jclient != nullptr, nullptr);
 - 
 -     if (JackPortState* const port = jclient->portIdMapping[port_id])
 -         return (jack_port_t*)port;
 - 
 -     carla_stderr2("jack_port_by_id: invalid port id %u", port_id);
 -     return nullptr;
 - }
 - 
 - // --------------------------------------------------------------------------------------------------------------------
 - 
 - CARLA_PLUGIN_EXPORT
 - const char** jack_port_get_connections(const jack_port_t* port)
 - {
 -     carla_stderr2("%s(%p)", __FUNCTION__, port);
 - 
 -     const JackPortState* const jport = (const JackPortState*)port;
 -     CARLA_SAFE_ASSERT_RETURN(jport != nullptr, nullptr);
 -     CARLA_SAFE_ASSERT_RETURN(! jport->isSystem, nullptr);
 - 
 -     if (! jport->isConnected)
 -         return nullptr;
 - 
 -     return nullptr;
 - }
 - 
 - CARLA_PLUGIN_EXPORT
 - const char** jack_port_get_all_connections(const jack_client_t* client, const jack_port_t* port)
 - {
 -     carla_stdout("%s(%p, %p) WIP", __FUNCTION__, client, port);
 - 
 -     const JackClientState* const jclient = (const JackClientState*)client;
 -     CARLA_SAFE_ASSERT_RETURN(jclient != nullptr, nullptr);
 - 
 -     const JackPortState* const jport = (const JackPortState*)port;
 -     CARLA_SAFE_ASSERT_RETURN(jport != nullptr, nullptr);
 -     CARLA_SAFE_ASSERT_UINT_RETURN(jport->gid >= JackPortState::kPortIdOffsetAudioIn, jport->gid, nullptr);
 - 
 -     if (! jport->isConnected)
 -         return nullptr;
 - 
 -     if (jport->isSystem)
 -     {
 -         const JackPortState* connectedPort;
 - 
 -         /**/ if (jport->gid >= JackPortState::kPortIdOffsetMidiOut)
 -             connectedPort = jclient->midiOuts.getAt(jport->gid - JackPortState::kPortIdOffsetMidiOut, nullptr);
 -         else if (jport->gid >= JackPortState::kPortIdOffsetAudioOut)
 -             connectedPort = jclient->audioOuts.getAt(jport->gid - JackPortState::kPortIdOffsetAudioOut, nullptr);
 -         else if (jport->gid >= JackPortState::kPortIdOffsetMidiIn)
 -             connectedPort = jclient->midiIns.getAt(jport->gid - JackPortState::kPortIdOffsetMidiIn, nullptr);
 -         else
 -             connectedPort = jclient->audioIns.getAt(jport->gid - JackPortState::kPortIdOffsetAudioIn, nullptr);
 - 
 -         if (connectedPort == nullptr)
 -         {
 -             carla_debug("port %s has no connections?", jport->fullname);
 -             return nullptr;
 -         }
 - 
 -         if (const char** const ret = static_cast<const char**>(malloc(sizeof(const char*)*2)))
 -         {
 -             carla_debug("port %s is connected to %s", jport->fullname, connectedPort->fullname);
 -             ret[0] = connectedPort->fullname;
 -             ret[1] = nullptr;
 -             return ret;
 -         }
 -     }
 -     else
 -     {
 -         const JackServerState& jserver(jclient->server);
 -         const char* connectedPortName = nullptr;
 - 
 -         if (jport->isMidi)
 -         {
 -             if (jport->flags & JackPortIsOutput)
 -             {
 -                 if (jport->index < jserver.numMidiOuts)
 -                      connectedPortName = allocate_port_name("system:midi_playback_", jport->index);
 -             }
 -             else
 -             {
 -                 if (jport->index < jserver.numMidiIns)
 -                      connectedPortName = allocate_port_name("system:midi_capture_", jport->index);
 -             }
 -         }
 -         else
 -         {
 -             if (jport->flags & JackPortIsOutput)
 -             {
 -                 if (jport->index < jserver.numAudioOuts)
 -                     connectedPortName = allocate_port_name("system:playback_", jport->index);
 -             }
 -             else
 -             {
 -                 if (jport->index < jserver.numAudioIns)
 -                     connectedPortName = allocate_port_name("system:capture_", jport->index);
 -             }
 -         }
 - 
 -         if (connectedPortName != nullptr)
 -         {
 -             if (const char** const ret = static_cast<const char**>(malloc(sizeof(const char*)*2)))
 -             {
 -                 carla_debug("port %s is connected to %s", jport->fullname, connectedPortName);
 -                 ret[0] = connectedPortName;
 -                 ret[1] = nullptr;
 -                 return ret;
 -             }
 -         }
 -     }
 - 
 -     return nullptr;
 - }
 - 
 - // --------------------------------------------------------------------------------------------------------------------
 
 
  |