|
- /*
- * Carla JACK API for external applications
- * Copyright (C) 2016-2020 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_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_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_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_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_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;
- }
-
- // --------------------------------------------------------------------------------------------------------------------
|