|
|
@@ -0,0 +1,608 @@ |
|
|
|
/* |
|
|
|
Copyright (C) 2014 Cédric Schieli |
|
|
|
|
|
|
|
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. |
|
|
|
|
|
|
|
*/ |
|
|
|
|
|
|
|
#include "JackCompilerDeps.h" |
|
|
|
#include "driver_interface.h" |
|
|
|
#include "JackEngineControl.h" |
|
|
|
#include "JackLockedEngine.h" |
|
|
|
#include "JackWaitCallbackDriver.h" |
|
|
|
#include "JackProxyDriver.h" |
|
|
|
|
|
|
|
using namespace std; |
|
|
|
|
|
|
|
namespace Jack |
|
|
|
{ |
|
|
|
JackProxyDriver::JackProxyDriver(const char* name, const char* alias, JackLockedEngine* engine, JackSynchro* table, |
|
|
|
const char* upstream, const char* promiscuous, |
|
|
|
char* client_name, bool auto_connect, bool auto_save) |
|
|
|
: JackRestarterDriver(name, alias, engine, table) |
|
|
|
{ |
|
|
|
jack_log("JackProxyDriver::JackProxyDriver upstream %s", upstream); |
|
|
|
|
|
|
|
assert(strlen(upstream) < JACK_CLIENT_NAME_SIZE); |
|
|
|
strcpy(fUpstream, upstream); |
|
|
|
|
|
|
|
assert(strlen(client_name) < JACK_CLIENT_NAME_SIZE); |
|
|
|
strcpy(fClientName, client_name); |
|
|
|
|
|
|
|
if (promiscuous) { |
|
|
|
fPromiscuous = strdup(promiscuous); |
|
|
|
} |
|
|
|
|
|
|
|
fAutoConnect = auto_connect; |
|
|
|
fAutoSave = auto_save; |
|
|
|
} |
|
|
|
|
|
|
|
JackProxyDriver::~JackProxyDriver() |
|
|
|
{ |
|
|
|
if (fHandle) { |
|
|
|
UnloadJackModule(fHandle); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
int JackProxyDriver::LoadClientLib() |
|
|
|
{ |
|
|
|
// Already loaded |
|
|
|
if (fHandle) { |
|
|
|
return 0; |
|
|
|
} |
|
|
|
fHandle = LoadJackModule(JACK_PROXY_CLIENT_LIB); |
|
|
|
if (!fHandle) { |
|
|
|
return -1; |
|
|
|
} |
|
|
|
LoadSymbols(); |
|
|
|
return 0; |
|
|
|
} |
|
|
|
|
|
|
|
//open, close, attach and detach------------------------------------------------------ |
|
|
|
|
|
|
|
int JackProxyDriver::Open(jack_nframes_t buffer_size, |
|
|
|
jack_nframes_t samplerate, |
|
|
|
bool capturing, |
|
|
|
bool playing, |
|
|
|
int inchannels, |
|
|
|
int outchannels, |
|
|
|
bool monitor, |
|
|
|
const char* capture_driver_name, |
|
|
|
const char* playback_driver_name, |
|
|
|
jack_nframes_t capture_latency, |
|
|
|
jack_nframes_t playback_latency) |
|
|
|
{ |
|
|
|
fDetectPlaybackChannels = (outchannels == -1); |
|
|
|
fDetectCaptureChannels = (inchannels == -1); |
|
|
|
|
|
|
|
if (LoadClientLib() != 0) { |
|
|
|
jack_error("Cannot dynamically load client library !"); |
|
|
|
return -1; |
|
|
|
} |
|
|
|
|
|
|
|
return JackWaiterDriver::Open(buffer_size, samplerate, |
|
|
|
capturing, playing, |
|
|
|
inchannels, outchannels, |
|
|
|
monitor, |
|
|
|
capture_driver_name, playback_driver_name, |
|
|
|
capture_latency, playback_latency); |
|
|
|
} |
|
|
|
|
|
|
|
int JackProxyDriver::Close() |
|
|
|
{ |
|
|
|
FreePorts(); |
|
|
|
return JackWaiterDriver::Close(); |
|
|
|
} |
|
|
|
|
|
|
|
// Attach and Detach are defined as empty methods: port allocation is done when driver actually start (that is in Init) |
|
|
|
int JackProxyDriver::Attach() |
|
|
|
{ |
|
|
|
return 0; |
|
|
|
} |
|
|
|
|
|
|
|
int JackProxyDriver::Detach() |
|
|
|
{ |
|
|
|
return 0; |
|
|
|
} |
|
|
|
|
|
|
|
//init and restart-------------------------------------------------------------------- |
|
|
|
|
|
|
|
/* |
|
|
|
JackProxyDriver is wrapped in a JackWaitCallbackDriver decorator that behaves |
|
|
|
as a "dummy driver, until Initialize method returns. |
|
|
|
*/ |
|
|
|
bool JackProxyDriver::Initialize() |
|
|
|
{ |
|
|
|
jack_log("JackProxyDriver::Initialize"); |
|
|
|
|
|
|
|
// save existing local connections if needed |
|
|
|
if (fAutoSave) { |
|
|
|
SaveConnections(0); |
|
|
|
} |
|
|
|
|
|
|
|
// new loading, but existing client, restart the driver |
|
|
|
if (fClient) { |
|
|
|
jack_info("JackProxyDriver restarting..."); |
|
|
|
jack_client_close(fClient); |
|
|
|
} |
|
|
|
FreePorts(); |
|
|
|
|
|
|
|
// display some additional infos |
|
|
|
jack_info("JackProxyDriver started in %s mode.", |
|
|
|
(fEngineControl->fSyncMode) ? "sync" : "async"); |
|
|
|
|
|
|
|
do { |
|
|
|
jack_status_t status; |
|
|
|
char *old = NULL; |
|
|
|
|
|
|
|
if (fPromiscuous) { |
|
|
|
// as we are fiddling with the environment variable content, save it |
|
|
|
const char* tmp = getenv("JACK_PROMISCUOUS_SERVER"); |
|
|
|
if (tmp) { |
|
|
|
old = strdup(tmp); |
|
|
|
} |
|
|
|
// temporary enable promiscuous mode |
|
|
|
if (setenv("JACK_PROMISCUOUS_SERVER", fPromiscuous, 1) < 0) { |
|
|
|
free(old); |
|
|
|
jack_error("Error allocating memory."); |
|
|
|
return false; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
jack_info("JackProxyDriver connecting to %s", fUpstream); |
|
|
|
fClient = jack_client_open(fClientName, static_cast<jack_options_t>(JackNoStartServer|JackServerName), &status, fUpstream); |
|
|
|
|
|
|
|
if (fPromiscuous) { |
|
|
|
// restore previous environment variable content |
|
|
|
if (old) { |
|
|
|
if (setenv("JACK_PROMISCUOUS_SERVER", old, 1) < 0) { |
|
|
|
free(old); |
|
|
|
jack_error("Error allocating memory."); |
|
|
|
return false; |
|
|
|
} |
|
|
|
free(old); |
|
|
|
} else { |
|
|
|
unsetenv("JACK_PROMISCUOUS_SERVER"); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
// the connection failed, try again later |
|
|
|
if (!fClient) { |
|
|
|
JackSleep(1000000); |
|
|
|
} |
|
|
|
|
|
|
|
} while (!fClient); |
|
|
|
jack_info("JackProxyDriver connected to %s", fUpstream); |
|
|
|
|
|
|
|
// we are connected, let's register some callbacks |
|
|
|
|
|
|
|
jack_on_shutdown(fClient, shutdown_callback, this); |
|
|
|
|
|
|
|
if (jack_set_process_callback(fClient, process_callback, this) != 0) { |
|
|
|
jack_error("Cannot set process callback."); |
|
|
|
return false; |
|
|
|
} |
|
|
|
|
|
|
|
if (jack_set_buffer_size_callback(fClient, bufsize_callback, this) != 0) { |
|
|
|
jack_error("Cannot set buffer size callback."); |
|
|
|
return false; |
|
|
|
} |
|
|
|
|
|
|
|
if (jack_set_sample_rate_callback(fClient, srate_callback, this) != 0) { |
|
|
|
jack_error("Cannot set sample rate callback."); |
|
|
|
return false; |
|
|
|
} |
|
|
|
|
|
|
|
if (jack_set_port_connect_callback(fClient, connect_callback, this) != 0) { |
|
|
|
jack_error("Cannot set port connect callback."); |
|
|
|
return false; |
|
|
|
} |
|
|
|
|
|
|
|
// detect upstream physical playback ports if needed |
|
|
|
if (fDetectPlaybackChannels) { |
|
|
|
fPlaybackChannels = CountIO(JACK_DEFAULT_AUDIO_TYPE, JackPortIsPhysical | JackPortIsOutput); |
|
|
|
} |
|
|
|
|
|
|
|
// detect upstream physical capture ports if needed |
|
|
|
if (fDetectCaptureChannels) { |
|
|
|
fCaptureChannels = CountIO(JACK_DEFAULT_AUDIO_TYPE, JackPortIsPhysical | JackPortIsInput); |
|
|
|
} |
|
|
|
|
|
|
|
if (AllocPorts() != 0) { |
|
|
|
jack_error("Can't allocate ports."); |
|
|
|
return false; |
|
|
|
} |
|
|
|
|
|
|
|
bufsize_callback(jack_get_buffer_size(fClient)); |
|
|
|
srate_callback(jack_get_sample_rate(fClient)); |
|
|
|
|
|
|
|
// restore local connections if needed |
|
|
|
if (fAutoSave) { |
|
|
|
LoadConnections(0); |
|
|
|
} |
|
|
|
|
|
|
|
// everything is ready, start upstream processing |
|
|
|
if (jack_activate(fClient) != 0) { |
|
|
|
jack_error("Cannot activate jack client."); |
|
|
|
return false; |
|
|
|
} |
|
|
|
|
|
|
|
// connect upstream ports if needed |
|
|
|
if (fAutoConnect) { |
|
|
|
ConnectPorts(); |
|
|
|
} |
|
|
|
|
|
|
|
return true; |
|
|
|
} |
|
|
|
|
|
|
|
int JackProxyDriver::Stop() |
|
|
|
{ |
|
|
|
if (fClient && (jack_deactivate(fClient) != 0)) { |
|
|
|
jack_error("Cannot deactivate jack client."); |
|
|
|
return -1; |
|
|
|
} |
|
|
|
return 0; |
|
|
|
} |
|
|
|
|
|
|
|
//client callbacks--------------------------------------------------------------------------- |
|
|
|
|
|
|
|
int JackProxyDriver::process_callback(jack_nframes_t nframes, void* arg) |
|
|
|
{ |
|
|
|
assert(static_cast<JackProxyDriver*>(arg)); |
|
|
|
return static_cast<JackProxyDriver*>(arg)->Process(); |
|
|
|
} |
|
|
|
|
|
|
|
int JackProxyDriver::bufsize_callback(jack_nframes_t nframes, void* arg) |
|
|
|
{ |
|
|
|
assert(static_cast<JackProxyDriver*>(arg)); |
|
|
|
return static_cast<JackProxyDriver*>(arg)->bufsize_callback(nframes); |
|
|
|
} |
|
|
|
int JackProxyDriver::bufsize_callback(jack_nframes_t nframes) |
|
|
|
{ |
|
|
|
if (JackTimedDriver::SetBufferSize(nframes) == 0) { |
|
|
|
return -1; |
|
|
|
} |
|
|
|
JackDriver::NotifyBufferSize(nframes); |
|
|
|
return 0; |
|
|
|
} |
|
|
|
|
|
|
|
int JackProxyDriver::srate_callback(jack_nframes_t nframes, void* arg) |
|
|
|
{ |
|
|
|
assert(static_cast<JackProxyDriver*>(arg)); |
|
|
|
return static_cast<JackProxyDriver*>(arg)->srate_callback(nframes); |
|
|
|
} |
|
|
|
int JackProxyDriver::srate_callback(jack_nframes_t nframes) |
|
|
|
{ |
|
|
|
if (JackTimedDriver::SetSampleRate(nframes) == 0) { |
|
|
|
return -1; |
|
|
|
} |
|
|
|
JackDriver::NotifySampleRate(nframes); |
|
|
|
return 0; |
|
|
|
} |
|
|
|
|
|
|
|
void JackProxyDriver::connect_callback(jack_port_id_t a, jack_port_id_t b, int connect, void* arg) |
|
|
|
{ |
|
|
|
assert(static_cast<JackProxyDriver*>(arg)); |
|
|
|
static_cast<JackProxyDriver*>(arg)->connect_callback(a, b, connect); |
|
|
|
} |
|
|
|
void JackProxyDriver::connect_callback(jack_port_id_t a, jack_port_id_t b, int connect) |
|
|
|
{ |
|
|
|
jack_port_t* port; |
|
|
|
int i; |
|
|
|
|
|
|
|
// skip port if not our own |
|
|
|
port = jack_port_by_id(fClient, a); |
|
|
|
if (!jack_port_is_mine(fClient, port)) { |
|
|
|
port = jack_port_by_id(fClient, b); |
|
|
|
if (!jack_port_is_mine(fClient, port)) { |
|
|
|
return; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
for (i = 0; i < fCaptureChannels; i++) { |
|
|
|
if (fUpstreamPlaybackPorts[i] == port) { |
|
|
|
fUpstreamPlaybackPortConnected[i] = connect; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
for (i = 0; i < fPlaybackChannels; i++) { |
|
|
|
if (fUpstreamCapturePorts[i] == port) { |
|
|
|
fUpstreamCapturePortConnected[i] = connect; |
|
|
|
} |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
void JackProxyDriver::shutdown_callback(void* arg) |
|
|
|
{ |
|
|
|
assert(static_cast<JackProxyDriver*>(arg)); |
|
|
|
static_cast<JackProxyDriver*>(arg)->RestartWait(); |
|
|
|
} |
|
|
|
|
|
|
|
//jack ports and buffers-------------------------------------------------------------- |
|
|
|
|
|
|
|
int JackProxyDriver::CountIO(const char* type, int flags) |
|
|
|
{ |
|
|
|
int count = 0; |
|
|
|
const char** ports = jack_get_ports(fClient, NULL, type, flags); |
|
|
|
if (ports != NULL) { |
|
|
|
while (ports[count]) { count++; } |
|
|
|
jack_free(ports); |
|
|
|
} |
|
|
|
return count; |
|
|
|
} |
|
|
|
|
|
|
|
int JackProxyDriver::AllocPorts() |
|
|
|
{ |
|
|
|
jack_log("JackProxyDriver::AllocPorts fBufferSize = %ld fSampleRate = %ld", fEngineControl->fBufferSize, fEngineControl->fSampleRate); |
|
|
|
|
|
|
|
char proxy[REAL_JACK_PORT_NAME_SIZE]; |
|
|
|
int i; |
|
|
|
|
|
|
|
fUpstreamPlaybackPorts = new jack_port_t* [fCaptureChannels]; |
|
|
|
fUpstreamPlaybackPortConnected = new int [fCaptureChannels]; |
|
|
|
for (i = 0; i < fCaptureChannels; i++) { |
|
|
|
snprintf(proxy, sizeof(proxy), "%s:to_client_%d", fClientName, i + 1); |
|
|
|
fUpstreamPlaybackPorts[i] = jack_port_register(fClient, proxy, JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput | JackPortIsTerminal, 0); |
|
|
|
if (fUpstreamPlaybackPorts[i] == NULL) { |
|
|
|
jack_error("driver: cannot register upstream port %s", proxy); |
|
|
|
return -1; |
|
|
|
} |
|
|
|
fUpstreamPlaybackPortConnected[i] = 0; |
|
|
|
} |
|
|
|
|
|
|
|
fUpstreamCapturePorts = new jack_port_t* [fPlaybackChannels]; |
|
|
|
fUpstreamCapturePortConnected = new int [fPlaybackChannels]; |
|
|
|
for (i = 0; i < fPlaybackChannels; i++) { |
|
|
|
snprintf(proxy, sizeof(proxy), "%s:from_client_%d", fClientName, i + 1); |
|
|
|
fUpstreamCapturePorts[i] = jack_port_register(fClient, proxy, JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput | JackPortIsTerminal, 0); |
|
|
|
if (fUpstreamCapturePorts[i] == NULL) { |
|
|
|
jack_error("driver: cannot register upstream port %s", proxy); |
|
|
|
return -1; |
|
|
|
} |
|
|
|
fUpstreamCapturePortConnected[i] = 0; |
|
|
|
} |
|
|
|
|
|
|
|
// local ports are registered here |
|
|
|
return JackAudioDriver::Attach(); |
|
|
|
} |
|
|
|
|
|
|
|
int JackProxyDriver::FreePorts() |
|
|
|
{ |
|
|
|
jack_log("JackProxyDriver::FreePorts"); |
|
|
|
|
|
|
|
int i; |
|
|
|
|
|
|
|
for (i = 0; i < fCaptureChannels; i++) { |
|
|
|
if (fCapturePortList[i] > 0) { |
|
|
|
fEngine->PortUnRegister(fClientControl.fRefNum, fCapturePortList[i]); |
|
|
|
fCapturePortList[i] = 0; |
|
|
|
} |
|
|
|
if (fUpstreamPlaybackPorts && fUpstreamPlaybackPorts[i]) { |
|
|
|
fUpstreamPlaybackPorts[i] = NULL; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
for (i = 0; i < fPlaybackChannels; i++) { |
|
|
|
if (fPlaybackPortList[i] > 0) { |
|
|
|
fEngine->PortUnRegister(fClientControl.fRefNum, fPlaybackPortList[i]); |
|
|
|
fPlaybackPortList[i] = 0; |
|
|
|
} |
|
|
|
if (fUpstreamCapturePorts && fUpstreamCapturePorts[i]) { |
|
|
|
fUpstreamCapturePorts[i] = NULL; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
delete[] fUpstreamPlaybackPorts; |
|
|
|
delete[] fUpstreamPlaybackPortConnected; |
|
|
|
delete[] fUpstreamCapturePorts; |
|
|
|
delete[] fUpstreamCapturePortConnected; |
|
|
|
|
|
|
|
fUpstreamPlaybackPorts = NULL; |
|
|
|
fUpstreamPlaybackPortConnected = NULL; |
|
|
|
fUpstreamCapturePorts = NULL; |
|
|
|
fUpstreamCapturePortConnected = NULL; |
|
|
|
|
|
|
|
return 0; |
|
|
|
} |
|
|
|
|
|
|
|
void JackProxyDriver::ConnectPorts() |
|
|
|
{ |
|
|
|
jack_log("JackProxyDriver::ConnectPorts"); |
|
|
|
const char** ports = jack_get_ports(fClient, NULL, JACK_DEFAULT_AUDIO_TYPE, JackPortIsPhysical | JackPortIsOutput); |
|
|
|
if (ports != NULL) { |
|
|
|
for (int i = 0; i < fCaptureChannels && ports[i]; i++) { |
|
|
|
jack_connect(fClient, ports[i], jack_port_name(fUpstreamPlaybackPorts[i])); |
|
|
|
} |
|
|
|
jack_free(ports); |
|
|
|
} |
|
|
|
|
|
|
|
ports = jack_get_ports(fClient, NULL, JACK_DEFAULT_AUDIO_TYPE, JackPortIsPhysical | JackPortIsInput); |
|
|
|
if (ports != NULL) { |
|
|
|
for (int i = 0; i < fPlaybackChannels && ports[i]; i++) { |
|
|
|
jack_connect(fClient, jack_port_name(fUpstreamCapturePorts[i]), ports[i]); |
|
|
|
} |
|
|
|
jack_free(ports); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
//driver processes-------------------------------------------------------------------- |
|
|
|
|
|
|
|
int JackProxyDriver::Read() |
|
|
|
{ |
|
|
|
// take the time at the beginning of the cycle |
|
|
|
JackDriver::CycleTakeBeginTime(); |
|
|
|
|
|
|
|
int i; |
|
|
|
void *from, *to; |
|
|
|
size_t buflen = sizeof(jack_default_audio_sample_t) * fEngineControl->fBufferSize; |
|
|
|
|
|
|
|
for (i = 0; i < fCaptureChannels; i++) { |
|
|
|
if (fUpstreamPlaybackPortConnected[i]) { |
|
|
|
from = jack_port_get_buffer(fUpstreamPlaybackPorts[i], fEngineControl->fBufferSize); |
|
|
|
to = GetInputBuffer(i); |
|
|
|
memcpy(to, from, buflen); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
return 0; |
|
|
|
} |
|
|
|
|
|
|
|
int JackProxyDriver::Write() |
|
|
|
{ |
|
|
|
int i; |
|
|
|
void *from, *to; |
|
|
|
size_t buflen = sizeof(jack_default_audio_sample_t) * fEngineControl->fBufferSize; |
|
|
|
|
|
|
|
for (i = 0; i < fPlaybackChannels; i++) { |
|
|
|
if (fUpstreamCapturePortConnected[i]) { |
|
|
|
to = jack_port_get_buffer(fUpstreamCapturePorts[i], fEngineControl->fBufferSize); |
|
|
|
from = GetOutputBuffer(i); |
|
|
|
memcpy(to, from, buflen); |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
return 0; |
|
|
|
} |
|
|
|
|
|
|
|
//driver loader----------------------------------------------------------------------- |
|
|
|
|
|
|
|
#ifdef __cplusplus |
|
|
|
extern "C" |
|
|
|
{ |
|
|
|
#endif |
|
|
|
|
|
|
|
SERVER_EXPORT jack_driver_desc_t* driver_get_descriptor() |
|
|
|
{ |
|
|
|
jack_driver_desc_t * desc; |
|
|
|
jack_driver_desc_filler_t filler; |
|
|
|
jack_driver_param_value_t value; |
|
|
|
|
|
|
|
desc = jack_driver_descriptor_construct("proxy", JackDriverMaster, "proxy backend", &filler); |
|
|
|
|
|
|
|
strcpy(value.str, DEFAULT_UPSTREAM); |
|
|
|
jack_driver_descriptor_add_parameter(desc, &filler, "upstream", 'u', JackDriverParamString, &value, NULL, "Name of the upstream jack server", NULL); |
|
|
|
|
|
|
|
strcpy(value.str, ""); |
|
|
|
jack_driver_descriptor_add_parameter(desc, &filler, "promiscuous", 'p', JackDriverParamString, &value, NULL, "Promiscuous group", NULL); |
|
|
|
|
|
|
|
value.i = -1; |
|
|
|
jack_driver_descriptor_add_parameter(desc, &filler, "input-ports", 'C', JackDriverParamInt, &value, NULL, "Number of audio input ports", "Number of audio input ports. If -1, audio physical input from the master"); |
|
|
|
jack_driver_descriptor_add_parameter(desc, &filler, "output-ports", 'P', JackDriverParamInt, &value, NULL, "Number of audio output ports", "Number of audio output ports. If -1, audio physical output from the master"); |
|
|
|
|
|
|
|
strcpy(value.str, "proxy"); |
|
|
|
jack_driver_descriptor_add_parameter(desc, &filler, "client-name", 'n', JackDriverParamString, &value, NULL, "Name of the jack client", NULL); |
|
|
|
|
|
|
|
value.i = false; |
|
|
|
jack_driver_descriptor_add_parameter(desc, &filler, "use-username", 'U', JackDriverParamBool, &value, NULL, "Use current username as client name", NULL); |
|
|
|
|
|
|
|
value.i = false; |
|
|
|
jack_driver_descriptor_add_parameter(desc, &filler, "auto-connect", 'c', JackDriverParamBool, &value, NULL, "Auto connect proxy to upstream system ports", NULL); |
|
|
|
|
|
|
|
value.i = false; |
|
|
|
jack_driver_descriptor_add_parameter(desc, &filler, "auto-save", 's', JackDriverParamBool, &value, NULL, "Save/restore connection state when restarting", NULL); |
|
|
|
|
|
|
|
return desc; |
|
|
|
} |
|
|
|
|
|
|
|
SERVER_EXPORT Jack::JackDriverClientInterface* driver_initialize(Jack::JackLockedEngine* engine, Jack::JackSynchro* table, const JSList* params) |
|
|
|
{ |
|
|
|
char upstream[JACK_CLIENT_NAME_SIZE + 1]; |
|
|
|
char promiscuous[JACK_CLIENT_NAME_SIZE + 1] = {0}; |
|
|
|
char client_name[JACK_CLIENT_NAME_SIZE + 1]; |
|
|
|
jack_nframes_t period_size = 1024; // to be used while waiting for master period_size |
|
|
|
jack_nframes_t sample_rate = 48000; // to be used while waiting for master sample_rate |
|
|
|
int capture_ports = -1; |
|
|
|
int playback_ports = -1; |
|
|
|
const JSList* node; |
|
|
|
const jack_driver_param_t* param; |
|
|
|
bool auto_connect = false; |
|
|
|
bool auto_save = false; |
|
|
|
bool use_promiscuous = false; |
|
|
|
|
|
|
|
// Possibly use env variable for upstream name |
|
|
|
const char* default_upstream = getenv("JACK_PROXY_UPSTREAM"); |
|
|
|
strcpy(upstream, (default_upstream) ? default_upstream : DEFAULT_UPSTREAM); |
|
|
|
|
|
|
|
// Possibly use env variable for upstream promiscuous |
|
|
|
const char* default_promiscuous = getenv("JACK_PROXY_PROMISCUOUS"); |
|
|
|
strcpy(promiscuous, (default_promiscuous) ? default_promiscuous : ""); |
|
|
|
|
|
|
|
// Possibly use env variable for client name |
|
|
|
const char* default_client_name = getenv("JACK_PROXY_CLIENT_NAME"); |
|
|
|
strcpy(client_name, (default_client_name) ? default_client_name : DEFAULT_CLIENT_NAME); |
|
|
|
|
|
|
|
#ifdef WIN32 |
|
|
|
const char* username = getenv("USERNAME"); |
|
|
|
#else |
|
|
|
const char* username = getenv("LOGNAME"); |
|
|
|
#endif |
|
|
|
|
|
|
|
for (node = params; node; node = jack_slist_next(node)) { |
|
|
|
param = (const jack_driver_param_t*) node->data; |
|
|
|
switch (param->character) |
|
|
|
{ |
|
|
|
case 'u' : |
|
|
|
assert(strlen(param->value.str) < JACK_CLIENT_NAME_SIZE); |
|
|
|
strcpy(upstream, param->value.str); |
|
|
|
break; |
|
|
|
case 'p': |
|
|
|
assert(strlen(param->value.str) < JACK_CLIENT_NAME_SIZE); |
|
|
|
use_promiscuous = true; |
|
|
|
strcpy(promiscuous, param->value.str); |
|
|
|
break; |
|
|
|
case 'C': |
|
|
|
capture_ports = param->value.i; |
|
|
|
break; |
|
|
|
case 'P': |
|
|
|
playback_ports = param->value.i; |
|
|
|
break; |
|
|
|
case 'n' : |
|
|
|
assert(strlen(param->value.str) < JACK_CLIENT_NAME_SIZE); |
|
|
|
strncpy(client_name, param->value.str, JACK_CLIENT_NAME_SIZE); |
|
|
|
break; |
|
|
|
case 'U' : |
|
|
|
if (username && *username) { |
|
|
|
assert(strlen(username) < JACK_CLIENT_NAME_SIZE); |
|
|
|
strncpy(client_name, username, JACK_CLIENT_NAME_SIZE); |
|
|
|
} |
|
|
|
case 'c': |
|
|
|
auto_connect = true; |
|
|
|
break; |
|
|
|
case 's': |
|
|
|
auto_save = true; |
|
|
|
break; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
try { |
|
|
|
|
|
|
|
Jack::JackDriverClientInterface* driver = new Jack::JackWaitCallbackDriver( |
|
|
|
new Jack::JackProxyDriver("system", "proxy_pcm", engine, table, upstream, use_promiscuous ? promiscuous : NULL, client_name, auto_connect, auto_save)); |
|
|
|
if (driver->Open(period_size, sample_rate, 1, 1, capture_ports, playback_ports, false, "capture_", "playback_", 0, 0) == 0) { |
|
|
|
return driver; |
|
|
|
} else { |
|
|
|
delete driver; |
|
|
|
return NULL; |
|
|
|
} |
|
|
|
|
|
|
|
} catch (...) { |
|
|
|
return NULL; |
|
|
|
} |
|
|
|
} |
|
|
|
|
|
|
|
#ifdef __cplusplus |
|
|
|
} |
|
|
|
#endif |
|
|
|
} |