jack2 codebase
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

1011 lines
37KB

  1. /*
  2. Copyright(C) 2008-2011 Romain Moret at Grame
  3. This program is free software; you can redistribute it and/or modify
  4. it under the terms of the GNU General Public License as published by
  5. the Free Software Foundation; either version 2 of the License, or
  6. (at your option) any later version.
  7. This program is distributed in the hope that it will be useful,
  8. but WITHOUT ANY WARRANTY; without even the implied warranty of
  9. MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  10. GNU General Public License for more details.
  11. You should have received a copy of the GNU General Public License
  12. along with this program; if not, write to the Free Software
  13. Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
  14. */
  15. #include "JackNetManager.h"
  16. #include "JackArgParser.h"
  17. #include "JackServerGlobals.h"
  18. #include "JackLockedEngine.h"
  19. #include "thread.h"
  20. #ifdef _WIN32
  21. #include "JackMetadata.h"
  22. #include "JackWinNamedPipe.h"
  23. #include "JackNetInterface.h"
  24. #endif
  25. namespace Jack
  26. {
  27. //JackNetMaster******************************************************************************************************
  28. JackNetMaster::JackNetMaster(JackNetSocket& socket, session_params_t& params, const char* multicast_ip)
  29. : JackNetMasterInterface(params, socket, multicast_ip)
  30. {
  31. jack_log("JackNetMaster::JackNetMaster");
  32. //settings
  33. fName = const_cast<char*>(fParams.fName);
  34. fClient = NULL;
  35. fSendTransportData.fState = -1;
  36. fReturnTransportData.fState = -1;
  37. fLastTransportState = -1;
  38. int port_index;
  39. //jack audio ports
  40. fAudioCapturePorts = new jack_port_t* [fParams.fSendAudioChannels];
  41. for (port_index = 0; port_index < fParams.fSendAudioChannels; port_index++) {
  42. fAudioCapturePorts[port_index] = NULL;
  43. }
  44. fAudioPlaybackPorts = new jack_port_t* [fParams.fReturnAudioChannels];
  45. for (port_index = 0; port_index < fParams.fReturnAudioChannels; port_index++) {
  46. fAudioPlaybackPorts[port_index] = NULL;
  47. }
  48. //jack midi ports
  49. fMidiCapturePorts = new jack_port_t* [fParams.fSendMidiChannels];
  50. for (port_index = 0; port_index < fParams.fSendMidiChannels; port_index++) {
  51. fMidiCapturePorts[port_index] = NULL;
  52. }
  53. fMidiPlaybackPorts = new jack_port_t* [fParams.fReturnMidiChannels];
  54. for (port_index = 0; port_index < fParams.fReturnMidiChannels; port_index++) {
  55. fMidiPlaybackPorts[port_index] = NULL;
  56. }
  57. //monitor
  58. #ifdef JACK_MONITOR
  59. fPeriodUsecs = (int)(1000000.f * ((float) fParams.fPeriodSize / (float) fParams.fSampleRate));
  60. string plot_name;
  61. plot_name = string(fParams.fName);
  62. plot_name += string("_master");
  63. plot_name += string((fParams.fSlaveSyncMode) ? "_sync" : "_async");
  64. plot_name += string("_latency");
  65. fNetTimeMon = new JackGnuPlotMonitor<float>(128, 4, plot_name);
  66. string net_time_mon_fields[] =
  67. {
  68. string("sync send"),
  69. string("end of send"),
  70. string("sync recv"),
  71. string("end of cycle")
  72. };
  73. string net_time_mon_options[] =
  74. {
  75. string("set xlabel \"audio cycles\""),
  76. string("set ylabel \"% of audio cycle\"")
  77. };
  78. fNetTimeMon->SetPlotFile(net_time_mon_options, 2, net_time_mon_fields, 4);
  79. #endif
  80. }
  81. JackNetMaster::~JackNetMaster()
  82. {
  83. jack_log("JackNetMaster::~JackNetMaster ID = %u", fParams.fID);
  84. if (fClient) {
  85. jack_deactivate(fClient);
  86. FreePorts();
  87. jack_client_close(fClient);
  88. }
  89. delete[] fAudioCapturePorts;
  90. delete[] fAudioPlaybackPorts;
  91. delete[] fMidiCapturePorts;
  92. delete[] fMidiPlaybackPorts;
  93. #ifdef JACK_MONITOR
  94. fNetTimeMon->Save();
  95. delete fNetTimeMon;
  96. #endif
  97. }
  98. //init--------------------------------------------------------------------------------
  99. bool JackNetMaster::Init(bool auto_connect)
  100. {
  101. //network init
  102. if (!JackNetMasterInterface::Init()) {
  103. jack_error("JackNetMasterInterface::Init() error...");
  104. return false;
  105. }
  106. //set global parameters
  107. if (!SetParams()) {
  108. jack_error("SetParams error...");
  109. return false;
  110. }
  111. //jack client and process
  112. jack_status_t status;
  113. if ((fClient = jack_client_open(fName, JackNullOption, &status, NULL)) == NULL) {
  114. jack_error("Can't open a new JACK client");
  115. return false;
  116. }
  117. if (jack_set_process_callback(fClient, SetProcess, this) < 0) {
  118. goto fail;
  119. }
  120. if (jack_set_buffer_size_callback(fClient, SetBufferSize, this) < 0) {
  121. goto fail;
  122. }
  123. if (jack_set_sample_rate_callback(fClient, SetSampleRate, this) < 0) {
  124. goto fail;
  125. }
  126. if (jack_set_latency_callback(fClient, LatencyCallback, this) < 0) {
  127. goto fail;
  128. }
  129. /*
  130. if (jack_set_port_connect_callback(fClient, SetConnectCallback, this) < 0) {
  131. goto fail;
  132. }
  133. */
  134. if (AllocPorts() != 0) {
  135. jack_error("Can't allocate JACK ports");
  136. goto fail;
  137. }
  138. //process can now run
  139. fRunning = true;
  140. //finally activate jack client
  141. if (jack_activate(fClient) != 0) {
  142. jack_error("Can't activate JACK client");
  143. goto fail;
  144. }
  145. if (auto_connect) {
  146. ConnectPorts();
  147. }
  148. jack_info("New NetMaster started");
  149. return true;
  150. fail:
  151. FreePorts();
  152. jack_client_close(fClient);
  153. fClient = NULL;
  154. return false;
  155. }
  156. //jack ports--------------------------------------------------------------------------
  157. int JackNetMaster::AllocPorts()
  158. {
  159. int i;
  160. char name[32];
  161. jack_log("JackNetMaster::AllocPorts");
  162. //audio
  163. for (i = 0; i < fParams.fSendAudioChannels; i++) {
  164. snprintf(name, sizeof(name), "to_slave_%d", i+1);
  165. if ((fAudioCapturePorts[i] = jack_port_register(fClient, name, JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput | JackPortIsTerminal, 0)) == NULL) {
  166. return -1;
  167. }
  168. }
  169. for (i = 0; i < fParams.fReturnAudioChannels; i++) {
  170. snprintf(name, sizeof(name), "from_slave_%d", i+1);
  171. if ((fAudioPlaybackPorts[i] = jack_port_register(fClient, name, JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput | JackPortIsTerminal, 0)) == NULL) {
  172. return -1;
  173. }
  174. }
  175. //midi
  176. for (i = 0; i < fParams.fSendMidiChannels; i++) {
  177. snprintf(name, sizeof(name), "midi_to_slave_%d", i+1);
  178. if ((fMidiCapturePorts[i] = jack_port_register(fClient, name, JACK_DEFAULT_MIDI_TYPE, JackPortIsInput | JackPortIsTerminal, 0)) == NULL) {
  179. return -1;
  180. }
  181. }
  182. for (i = 0; i < fParams.fReturnMidiChannels; i++) {
  183. snprintf(name, sizeof(name), "midi_from_slave_%d", i+1);
  184. if ((fMidiPlaybackPorts[i] = jack_port_register(fClient, name, JACK_DEFAULT_MIDI_TYPE, JackPortIsOutput | JackPortIsTerminal, 0)) == NULL) {
  185. return -1;
  186. }
  187. }
  188. return 0;
  189. }
  190. void JackNetMaster::ConnectPorts()
  191. {
  192. const char** ports = jack_get_ports(fClient, NULL, JACK_DEFAULT_AUDIO_TYPE, JackPortIsPhysical | JackPortIsOutput);
  193. if (ports != NULL) {
  194. for (int i = 0; i < fParams.fSendAudioChannels && ports[i]; i++) {
  195. jack_connect(fClient, ports[i], jack_port_name(fAudioCapturePorts[i]));
  196. }
  197. jack_free(ports);
  198. }
  199. ports = jack_get_ports(fClient, NULL, JACK_DEFAULT_AUDIO_TYPE, JackPortIsPhysical | JackPortIsInput);
  200. if (ports != NULL) {
  201. for (int i = 0; i < fParams.fReturnAudioChannels && ports[i]; i++) {
  202. jack_connect(fClient, jack_port_name(fAudioPlaybackPorts[i]), ports[i]);
  203. }
  204. jack_free(ports);
  205. }
  206. }
  207. void JackNetMaster::FreePorts()
  208. {
  209. jack_log("JackNetMaster::FreePorts ID = %u", fParams.fID);
  210. int port_index;
  211. for (port_index = 0; port_index < fParams.fSendAudioChannels; port_index++) {
  212. if (fAudioCapturePorts[port_index]) {
  213. jack_port_unregister(fClient, fAudioCapturePorts[port_index]);
  214. }
  215. }
  216. for (port_index = 0; port_index < fParams.fReturnAudioChannels; port_index++) {
  217. if (fAudioPlaybackPorts[port_index]) {
  218. jack_port_unregister(fClient, fAudioPlaybackPorts[port_index]);
  219. }
  220. }
  221. for (port_index = 0; port_index < fParams.fSendMidiChannels; port_index++) {
  222. if (fMidiCapturePorts[port_index]) {
  223. jack_port_unregister(fClient, fMidiCapturePorts[port_index]);
  224. }
  225. }
  226. for (port_index = 0; port_index < fParams.fReturnMidiChannels; port_index++) {
  227. if (fMidiPlaybackPorts[port_index]) {
  228. jack_port_unregister(fClient, fMidiPlaybackPorts[port_index]);
  229. }
  230. }
  231. }
  232. //transport---------------------------------------------------------------------------
  233. void JackNetMaster::EncodeTransportData()
  234. {
  235. //is there a new timebase master ?
  236. //TODO : check if any timebase callback has been called (and if it's conditional or not) and set correct value...
  237. fSendTransportData.fTimebaseMaster = NO_CHANGE;
  238. //update state and position
  239. fSendTransportData.fState = static_cast<uint>(jack_transport_query(fClient, &fSendTransportData.fPosition));
  240. //is it a new state ?
  241. fSendTransportData.fNewState = ((fSendTransportData.fState != fLastTransportState) && (fSendTransportData.fState != fReturnTransportData.fState));
  242. if (fSendTransportData.fNewState) {
  243. jack_info("Sending '%s' to '%s' frame = %ld", GetTransportState(fSendTransportData.fState), fParams.fName, fSendTransportData.fPosition.frame);
  244. }
  245. fLastTransportState = fSendTransportData.fState;
  246. }
  247. void JackNetMaster::DecodeTransportData()
  248. {
  249. //is there timebase master change ?
  250. if (fReturnTransportData.fTimebaseMaster != NO_CHANGE) {
  251. int timebase = 0;
  252. switch (fReturnTransportData.fTimebaseMaster)
  253. {
  254. case RELEASE_TIMEBASEMASTER :
  255. timebase = jack_release_timebase(fClient);
  256. if (timebase < 0) {
  257. jack_error("Can't release timebase master");
  258. } else {
  259. jack_info("'%s' isn't the timebase master anymore", fParams.fName);
  260. }
  261. break;
  262. case TIMEBASEMASTER :
  263. timebase = jack_set_timebase_callback(fClient, 0, SetTimebaseCallback, this);
  264. if (timebase < 0) {
  265. jack_error("Can't set a new timebase master");
  266. } else {
  267. jack_info("'%s' is the new timebase master", fParams.fName);
  268. }
  269. break;
  270. case CONDITIONAL_TIMEBASEMASTER :
  271. timebase = jack_set_timebase_callback(fClient, 1, SetTimebaseCallback, this);
  272. if (timebase != EBUSY) {
  273. if (timebase < 0)
  274. jack_error("Can't set a new timebase master");
  275. else
  276. jack_info("'%s' is the new timebase master", fParams.fName);
  277. }
  278. break;
  279. }
  280. }
  281. //is the slave in a new transport state and is this state different from master's ?
  282. if (fReturnTransportData.fNewState && (fReturnTransportData.fState != jack_transport_query(fClient, NULL))) {
  283. switch (fReturnTransportData.fState)
  284. {
  285. case JackTransportStopped :
  286. jack_transport_stop(fClient);
  287. jack_info("'%s' stops transport", fParams.fName);
  288. break;
  289. case JackTransportStarting :
  290. if (jack_transport_reposition(fClient, &fReturnTransportData.fPosition) == EINVAL)
  291. jack_error("Can't set new position");
  292. jack_transport_start(fClient);
  293. jack_info("'%s' starts transport frame = %d", fParams.fName, fReturnTransportData.fPosition.frame);
  294. break;
  295. case JackTransportNetStarting :
  296. jack_info("'%s' is ready to roll...", fParams.fName);
  297. break;
  298. case JackTransportRolling :
  299. jack_info("'%s' is rolling", fParams.fName);
  300. break;
  301. }
  302. }
  303. }
  304. void JackNetMaster::SetTimebaseCallback(jack_transport_state_t state, jack_nframes_t nframes, jack_position_t* pos, int new_pos, void* arg)
  305. {
  306. static_cast<JackNetMaster*>(arg)->TimebaseCallback(pos);
  307. }
  308. void JackNetMaster::TimebaseCallback(jack_position_t* pos)
  309. {
  310. pos->bar = fReturnTransportData.fPosition.bar;
  311. pos->beat = fReturnTransportData.fPosition.beat;
  312. pos->tick = fReturnTransportData.fPosition.tick;
  313. pos->bar_start_tick = fReturnTransportData.fPosition.bar_start_tick;
  314. pos->beats_per_bar = fReturnTransportData.fPosition.beats_per_bar;
  315. pos->beat_type = fReturnTransportData.fPosition.beat_type;
  316. pos->ticks_per_beat = fReturnTransportData.fPosition.ticks_per_beat;
  317. pos->beats_per_minute = fReturnTransportData.fPosition.beats_per_minute;
  318. }
  319. //sync--------------------------------------------------------------------------------
  320. bool JackNetMaster::IsSlaveReadyToRoll()
  321. {
  322. return (fReturnTransportData.fState == JackTransportNetStarting);
  323. }
  324. int JackNetMaster::SetBufferSize(jack_nframes_t nframes, void* arg)
  325. {
  326. JackNetMaster* obj = static_cast<JackNetMaster*>(arg);
  327. if (nframes != obj->fParams.fPeriodSize) {
  328. jack_error("Cannot currently handle buffer size change, so JackNetMaster proxy will be removed...");
  329. obj->Exit();
  330. }
  331. return 0;
  332. }
  333. int JackNetMaster::SetSampleRate(jack_nframes_t nframes, void* arg)
  334. {
  335. JackNetMaster* obj = static_cast<JackNetMaster*>(arg);
  336. if (nframes != obj->fParams.fSampleRate) {
  337. jack_error("Cannot currently handle sample rate change, so JackNetMaster proxy will be removed...");
  338. obj->Exit();
  339. }
  340. return 0;
  341. }
  342. void JackNetMaster::LatencyCallback(jack_latency_callback_mode_t mode, void* arg)
  343. {
  344. JackNetMaster* obj = static_cast<JackNetMaster*>(arg);
  345. jack_nframes_t port_latency = jack_get_buffer_size(obj->fClient);
  346. jack_latency_range_t range;
  347. //audio
  348. for (int i = 0; i < obj->fParams.fSendAudioChannels; i++) {
  349. //port latency
  350. range.min = range.max = float(obj->fParams.fNetworkLatency * port_latency) / 2.f;
  351. jack_port_set_latency_range(obj->fAudioCapturePorts[i], JackPlaybackLatency, &range);
  352. }
  353. //audio
  354. for (int i = 0; i < obj->fParams.fReturnAudioChannels; i++) {
  355. //port latency
  356. range.min = range.max = float(obj->fParams.fNetworkLatency * port_latency) / 2.f + ((obj->fParams.fSlaveSyncMode) ? 0 : port_latency);
  357. jack_port_set_latency_range(obj->fAudioPlaybackPorts[i], JackCaptureLatency, &range);
  358. }
  359. //midi
  360. for (int i = 0; i < obj->fParams.fSendMidiChannels; i++) {
  361. //port latency
  362. range.min = range.max = float(obj->fParams.fNetworkLatency * port_latency) / 2.f;
  363. jack_port_set_latency_range(obj->fMidiCapturePorts[i], JackPlaybackLatency, &range);
  364. }
  365. //midi
  366. for (int i = 0; i < obj->fParams.fReturnMidiChannels; i++) {
  367. //port latency
  368. range.min = range.max = obj->fParams.fNetworkLatency * port_latency + ((obj->fParams.fSlaveSyncMode) ? 0 : port_latency);
  369. jack_port_set_latency_range(obj->fMidiPlaybackPorts[i], JackCaptureLatency, &range);
  370. }
  371. }
  372. //process-----------------------------------------------------------------------------
  373. int JackNetMaster::SetProcess(jack_nframes_t nframes, void* arg)
  374. {
  375. try {
  376. return static_cast<JackNetMaster*>(arg)->Process();
  377. } catch (JackNetException& e) {
  378. return 0;
  379. }
  380. }
  381. void JackNetMaster::SetConnectCallback(jack_port_id_t a, jack_port_id_t b, int connect, void* arg)
  382. {
  383. static_cast<JackNetMaster*>(arg)->ConnectCallback(a, b, connect);
  384. }
  385. void JackNetMaster::ConnectCallback(jack_port_id_t a, jack_port_id_t b, int connect)
  386. {
  387. jack_info("JackNetMaster::ConnectCallback a = %d b = %d connect = %d", a, b, connect);
  388. if (connect) {
  389. jack_connect(fClient, jack_port_name(jack_port_by_id(fClient, a)), "system:playback_1");
  390. }
  391. }
  392. int JackNetMaster::Process()
  393. {
  394. if (!fRunning) {
  395. return 0;
  396. }
  397. #ifdef JACK_MONITOR
  398. jack_time_t begin_time = GetMicroSeconds();
  399. fNetTimeMon->New();
  400. #endif
  401. //buffers
  402. for (int midi_port_index = 0; midi_port_index < fParams.fSendMidiChannels; midi_port_index++) {
  403. fNetMidiCaptureBuffer->SetBuffer(midi_port_index,
  404. static_cast<JackMidiBuffer*>(jack_port_get_buffer(fMidiCapturePorts[midi_port_index],
  405. fParams.fPeriodSize)));
  406. }
  407. for (int audio_port_index = 0; audio_port_index < fParams.fSendAudioChannels; audio_port_index++) {
  408. #ifdef OPTIMIZED_PROTOCOL
  409. if (fNetAudioCaptureBuffer->GetConnected(audio_port_index)) {
  410. // Port is connected on other side...
  411. fNetAudioCaptureBuffer->SetBuffer(audio_port_index,
  412. ((jack_port_connected(fAudioCapturePorts[audio_port_index]) > 0)
  413. ? static_cast<sample_t*>(jack_port_get_buffer(fAudioCapturePorts[audio_port_index], fParams.fPeriodSize))
  414. : NULL));
  415. } else {
  416. fNetAudioCaptureBuffer->SetBuffer(audio_port_index, NULL);
  417. }
  418. #else
  419. fNetAudioCaptureBuffer->SetBuffer(audio_port_index,
  420. static_cast<sample_t*>(jack_port_get_buffer(fAudioCapturePorts[audio_port_index],
  421. fParams.fPeriodSize)));
  422. #endif
  423. // TODO
  424. }
  425. for (int midi_port_index = 0; midi_port_index < fParams.fReturnMidiChannels; midi_port_index++) {
  426. fNetMidiPlaybackBuffer->SetBuffer(midi_port_index,
  427. static_cast<JackMidiBuffer*>(jack_port_get_buffer(fMidiPlaybackPorts[midi_port_index],
  428. fParams.fPeriodSize)));
  429. }
  430. for (int audio_port_index = 0; audio_port_index < fParams.fReturnAudioChannels; audio_port_index++) {
  431. #ifdef OPTIMIZED_PROTOCOL
  432. sample_t* out = (jack_port_connected(fAudioPlaybackPorts[audio_port_index]) > 0)
  433. ? static_cast<sample_t*>(jack_port_get_buffer(fAudioPlaybackPorts[audio_port_index], fParams.fPeriodSize))
  434. : NULL;
  435. if (out) {
  436. memset(out, 0, sizeof(float) * fParams.fPeriodSize);
  437. }
  438. fNetAudioPlaybackBuffer->SetBuffer(audio_port_index, out);
  439. #else
  440. sample_t* out = static_cast<sample_t*>(jack_port_get_buffer(fAudioPlaybackPorts[audio_port_index], fParams.fPeriodSize));
  441. if (out) {
  442. memset(out, 0, sizeof(float) * fParams.fPeriodSize);
  443. }
  444. fNetAudioPlaybackBuffer->SetBuffer(audio_port_index, out)));
  445. #endif
  446. }
  447. // encode the first packet
  448. EncodeSyncPacket();
  449. if (SyncSend() == SOCKET_ERROR) {
  450. return SOCKET_ERROR;
  451. }
  452. #ifdef JACK_MONITOR
  453. fNetTimeMon->Add((((float)(GetMicroSeconds() - begin_time)) / (float) fPeriodUsecs) * 100.f);
  454. #endif
  455. // send data
  456. if (DataSend() == SOCKET_ERROR) {
  457. return SOCKET_ERROR;
  458. }
  459. #ifdef JACK_MONITOR
  460. fNetTimeMon->Add((((float)(GetMicroSeconds() - begin_time)) / (float) fPeriodUsecs) * 100.f);
  461. #endif
  462. // receive sync
  463. int res = SyncRecv();
  464. switch (res) {
  465. case NET_SYNCHING:
  466. case SOCKET_ERROR:
  467. return res;
  468. case SYNC_PACKET_ERROR:
  469. // Since sync packet is incorrect, don't decode it and continue with data
  470. break;
  471. default:
  472. // Decode sync
  473. int unused_frames;
  474. DecodeSyncPacket(unused_frames);
  475. break;
  476. }
  477. #ifdef JACK_MONITOR
  478. fNetTimeMon->Add((((float)(GetMicroSeconds() - begin_time)) / (float) fPeriodUsecs) * 100.f);
  479. #endif
  480. // receive data
  481. res = DataRecv();
  482. switch (res) {
  483. case 0:
  484. case SOCKET_ERROR:
  485. return res;
  486. case DATA_PACKET_ERROR:
  487. // Well not a real XRun...
  488. JackServerGlobals::fInstance->GetEngine()->NotifyClientXRun(ALL_CLIENTS);
  489. break;
  490. }
  491. #ifdef JACK_MONITOR
  492. fNetTimeMon->AddLast((((float)(GetMicroSeconds() - begin_time)) / (float) fPeriodUsecs) * 100.f);
  493. #endif
  494. return 0;
  495. }
  496. void JackNetMaster::SaveConnections(connections_list_t& connections)
  497. {
  498. // Audio
  499. for (int i = 0; i < fParams.fSendAudioChannels; i++) {
  500. const char** connected_port = jack_port_get_all_connections(fClient, fAudioCapturePorts[i]);
  501. if (connected_port != NULL) {
  502. for (int port = 0; connected_port[port]; port++) {
  503. connections.push_back(make_pair(connected_port[port], jack_port_name(fAudioCapturePorts[i])));
  504. jack_log("INPUT %s ==> %s", connected_port[port], jack_port_name(fAudioCapturePorts[i]));
  505. }
  506. jack_free(connected_port);
  507. }
  508. }
  509. for (int i = 0; i < fParams.fReturnAudioChannels; i++) {
  510. const char** connected_port = jack_port_get_all_connections(fClient, fAudioPlaybackPorts[i]);
  511. if (connected_port != NULL) {
  512. for (int port = 0; connected_port[port]; port++) {
  513. connections.push_back(make_pair(jack_port_name(fAudioPlaybackPorts[i]), connected_port[port]));
  514. jack_log("OUTPUT %s ==> %s", jack_port_name(fAudioPlaybackPorts[i]), connected_port[port]);
  515. }
  516. jack_free(connected_port);
  517. }
  518. }
  519. // MIDI
  520. for (int i = 0; i < fParams.fSendMidiChannels; i++) {
  521. const char** connected_port = jack_port_get_all_connections(fClient, fMidiCapturePorts[i]);
  522. if (connected_port != NULL) {
  523. for (int port = 0; connected_port[port]; port++) {
  524. connections.push_back(make_pair(connected_port[port], jack_port_name(fMidiCapturePorts[i])));
  525. jack_log("INPUT %s ==> %s", connected_port[port], jack_port_name(fMidiCapturePorts[i]));
  526. }
  527. jack_free(connected_port);
  528. }
  529. }
  530. for (int i = 0; i < fParams.fReturnMidiChannels; i++) {
  531. const char** connected_port = jack_port_get_all_connections(fClient, fMidiPlaybackPorts[i]);
  532. if (connected_port != NULL) {
  533. for (int port = 0; connected_port[port]; port++) {
  534. connections.push_back(make_pair(jack_port_name(fMidiPlaybackPorts[i]), connected_port[port]));
  535. jack_log("OUTPUT %s ==> %s", jack_port_name(fMidiPlaybackPorts[i]), connected_port[port]);
  536. }
  537. jack_free(connected_port);
  538. }
  539. }
  540. }
  541. void JackNetMaster::LoadConnections(const connections_list_t& connections)
  542. {
  543. list<pair<string, string> >::const_iterator it;
  544. for (it = connections.begin(); it != connections.end(); it++) {
  545. pair<string, string> connection = *it;
  546. jack_connect(fClient, connection.first.c_str(), connection.second.c_str());
  547. }
  548. }
  549. //JackNetMasterManager***********************************************************************************************
  550. JackNetMasterManager::JackNetMasterManager(jack_client_t* client, const JSList* params) : fSocket()
  551. {
  552. jack_log("JackNetMasterManager::JackNetMasterManager");
  553. fClient = client;
  554. fName = jack_get_client_name(fClient);
  555. fGlobalID = 0;
  556. fRunning = true;
  557. fAutoConnect = false;
  558. fAutoSave = false;
  559. const JSList* node;
  560. const jack_driver_param_t* param;
  561. jack_on_shutdown(fClient, SetShutDown, this);
  562. // Possibly use env variable
  563. const char* default_udp_port = getenv("JACK_NETJACK_PORT");
  564. fSocket.SetPort((default_udp_port) ? atoi(default_udp_port) : DEFAULT_PORT);
  565. const char* default_multicast_ip = getenv("JACK_NETJACK_MULTICAST");
  566. if (default_multicast_ip) {
  567. strcpy(fMulticastIP, default_multicast_ip);
  568. } else {
  569. strcpy(fMulticastIP, DEFAULT_MULTICAST_IP);
  570. }
  571. for (node = params; node; node = jack_slist_next(node)) {
  572. param = (const jack_driver_param_t*) node->data;
  573. switch (param->character) {
  574. case 'a' :
  575. if (strlen(param->value.str) < 32) {
  576. strcpy(fMulticastIP, param->value.str);
  577. } else {
  578. jack_error("Can't use multicast address %s, using default %s", param->value.ui, DEFAULT_MULTICAST_IP);
  579. }
  580. break;
  581. case 'p':
  582. fSocket.SetPort(param->value.ui);
  583. break;
  584. case 'c':
  585. fAutoConnect = true;
  586. break;
  587. case 's':
  588. fAutoSave = true;
  589. break;
  590. }
  591. }
  592. //set sync callback
  593. jack_set_sync_callback(fClient, SetSyncCallback, this);
  594. //activate the client (for sync callback)
  595. if (jack_activate(fClient) != 0) {
  596. jack_error("Can't activate the NetManager client, transport disabled");
  597. }
  598. //launch the manager thread
  599. if (jack_client_create_thread(fClient, &fThread, 0, 0, NetManagerThread, this)) {
  600. jack_error("Can't create the NetManager control thread");
  601. }
  602. }
  603. JackNetMasterManager::~JackNetMasterManager()
  604. {
  605. jack_log("JackNetMasterManager::~JackNetMasterManager");
  606. ShutDown();
  607. }
  608. int JackNetMasterManager::CountIO(const char* type, int flags)
  609. {
  610. int count = 0;
  611. const char** ports = jack_get_ports(fClient, NULL, type, flags);
  612. if (ports != NULL) {
  613. while (ports[count]) { count++; }
  614. jack_free(ports);
  615. }
  616. return count;
  617. }
  618. void JackNetMasterManager::SetShutDown(void* arg)
  619. {
  620. static_cast<JackNetMasterManager*>(arg)->ShutDown();
  621. }
  622. void JackNetMasterManager::ShutDown()
  623. {
  624. jack_log("JackNetMasterManager::ShutDown");
  625. if (fRunning) {
  626. jack_client_kill_thread(fClient, fThread);
  627. fRunning = false;
  628. }
  629. master_list_t::iterator it;
  630. for (it = fMasterList.begin(); it != fMasterList.end(); it++) {
  631. delete (*it);
  632. }
  633. fMasterList.clear();
  634. fSocket.Close();
  635. SocketAPIEnd();
  636. }
  637. int JackNetMasterManager::SetSyncCallback(jack_transport_state_t state, jack_position_t* pos, void* arg)
  638. {
  639. return static_cast<JackNetMasterManager*>(arg)->SyncCallback(state, pos);
  640. }
  641. int JackNetMasterManager::SyncCallback(jack_transport_state_t state, jack_position_t* pos)
  642. {
  643. //check if each slave is ready to roll
  644. int res = 1;
  645. master_list_it_t it;
  646. for (it = fMasterList.begin(); it != fMasterList.end(); it++) {
  647. if (!(*it)->IsSlaveReadyToRoll()) {
  648. res = 0;
  649. }
  650. }
  651. jack_log("JackNetMasterManager::SyncCallback returns '%s'", (res) ? "true" : "false");
  652. return res;
  653. }
  654. void* JackNetMasterManager::NetManagerThread(void* arg)
  655. {
  656. JackNetMasterManager* master_manager = static_cast<JackNetMasterManager*>(arg);
  657. jack_info("Starting Jack NetManager");
  658. jack_info("Listening on '%s:%d'", master_manager->fMulticastIP, master_manager->fSocket.GetPort());
  659. master_manager->Run();
  660. return NULL;
  661. }
  662. void JackNetMasterManager::Run()
  663. {
  664. jack_log("JackNetMasterManager::Run");
  665. //utility variables
  666. int attempt = 0;
  667. //data
  668. session_params_t host_params;
  669. int rx_bytes = 0;
  670. JackNetMaster* net_master;
  671. //init socket API (win32)
  672. if (SocketAPIInit() < 0) {
  673. jack_error("Can't init Socket API, exiting...");
  674. return;
  675. }
  676. //socket
  677. if (fSocket.NewSocket() == SOCKET_ERROR) {
  678. jack_error("Can't create NetManager input socket : %s", StrError(NET_ERROR_CODE));
  679. return;
  680. }
  681. //bind the socket to the local port
  682. if (fSocket.Bind() == SOCKET_ERROR) {
  683. jack_error("Can't bind NetManager socket : %s", StrError(NET_ERROR_CODE));
  684. fSocket.Close();
  685. return;
  686. }
  687. //join multicast group
  688. if (fSocket.JoinMCastGroup(fMulticastIP) == SOCKET_ERROR) {
  689. jack_error("Can't join multicast group : %s", StrError(NET_ERROR_CODE));
  690. }
  691. //local loop
  692. if (fSocket.SetLocalLoop() == SOCKET_ERROR) {
  693. jack_error("Can't set local loop : %s", StrError(NET_ERROR_CODE));
  694. }
  695. //set a timeout on the multicast receive (the thread can now be cancelled)
  696. if (fSocket.SetTimeOut(MANAGER_INIT_TIMEOUT) == SOCKET_ERROR) {
  697. jack_error("Can't set timeout : %s", StrError(NET_ERROR_CODE));
  698. }
  699. //main loop, wait for data, deal with it and wait again
  700. do
  701. {
  702. session_params_t net_params;
  703. rx_bytes = fSocket.CatchHost(&net_params, sizeof(session_params_t), 0);
  704. SessionParamsNToH(&net_params, &host_params);
  705. if ((rx_bytes == SOCKET_ERROR) && (fSocket.GetError() != NET_NO_DATA)) {
  706. jack_error("Error in receive : %s", StrError(NET_ERROR_CODE));
  707. if (++attempt == 10) {
  708. jack_error("Can't receive on the socket, exiting net manager");
  709. return;
  710. }
  711. }
  712. if (rx_bytes == sizeof(session_params_t)) {
  713. switch (GetPacketType(&host_params))
  714. {
  715. case SLAVE_AVAILABLE:
  716. if ((net_master = InitMaster(host_params))) {
  717. SessionParamsDisplay(&net_master->fParams);
  718. } else {
  719. jack_error("Can't init new NetMaster...");
  720. }
  721. jack_info("Waiting for a slave...");
  722. break;
  723. case KILL_MASTER:
  724. if (KillMaster(&host_params)) {
  725. jack_info("Waiting for a slave...");
  726. }
  727. break;
  728. default:
  729. break;
  730. }
  731. }
  732. }
  733. while (fRunning);
  734. }
  735. JackNetMaster* JackNetMasterManager::InitMaster(session_params_t& params)
  736. {
  737. jack_log("JackNetMasterManager::InitMaster slave : %s", params.fName);
  738. //check MASTER <<==> SLAVE network protocol coherency
  739. if (params.fProtocolVersion != NETWORK_PROTOCOL) {
  740. jack_error("Error : slave '%s' is running with a different protocol %d != %d", params.fName, params.fProtocolVersion, NETWORK_PROTOCOL);
  741. return NULL;
  742. }
  743. //settings
  744. fSocket.GetName(params.fMasterNetName);
  745. params.fID = ++fGlobalID;
  746. params.fSampleRate = jack_get_sample_rate(fClient);
  747. params.fPeriodSize = jack_get_buffer_size(fClient);
  748. if (params.fSendAudioChannels == -1) {
  749. params.fSendAudioChannels = CountIO(JACK_DEFAULT_AUDIO_TYPE, JackPortIsPhysical | JackPortIsOutput);
  750. jack_info("Takes physical %d audio input(s) for slave", params.fSendAudioChannels);
  751. }
  752. if (params.fReturnAudioChannels == -1) {
  753. params.fReturnAudioChannels = CountIO(JACK_DEFAULT_AUDIO_TYPE, JackPortIsPhysical | JackPortIsInput);
  754. jack_info("Takes physical %d audio output(s) for slave", params.fReturnAudioChannels);
  755. }
  756. if (params.fSendMidiChannels == -1) {
  757. params.fSendMidiChannels = CountIO(JACK_DEFAULT_MIDI_TYPE, JackPortIsPhysical | JackPortIsOutput);
  758. jack_info("Takes physical %d MIDI input(s) for slave", params.fSendMidiChannels);
  759. }
  760. if (params.fReturnMidiChannels == -1) {
  761. params.fReturnMidiChannels = CountIO(JACK_DEFAULT_MIDI_TYPE, JackPortIsPhysical | JackPortIsInput);
  762. jack_info("Takes physical %d MIDI output(s) for slave", params.fReturnMidiChannels);
  763. }
  764. //create a new master and add it to the list
  765. JackNetMaster* master = new JackNetMaster(fSocket, params, fMulticastIP);
  766. if (master->Init(fAutoConnect)) {
  767. fMasterList.push_back(master);
  768. if (fAutoSave && fMasterConnectionList.find(params.fName) != fMasterConnectionList.end()) {
  769. master->LoadConnections(fMasterConnectionList[params.fName]);
  770. }
  771. return master;
  772. } else {
  773. delete master;
  774. return NULL;
  775. }
  776. }
  777. master_list_it_t JackNetMasterManager::FindMaster(uint32_t id)
  778. {
  779. jack_log("JackNetMasterManager::FindMaster ID = %u", id);
  780. master_list_it_t it;
  781. for (it = fMasterList.begin(); it != fMasterList.end(); it++) {
  782. if ((*it)->fParams.fID == id) {
  783. return it;
  784. }
  785. }
  786. return it;
  787. }
  788. int JackNetMasterManager::KillMaster(session_params_t* params)
  789. {
  790. jack_log("JackNetMasterManager::KillMaster ID = %u", params->fID);
  791. master_list_it_t master_it = FindMaster(params->fID);
  792. if (master_it != fMasterList.end()) {
  793. if (fAutoSave) {
  794. fMasterConnectionList[params->fName].clear();
  795. (*master_it)->SaveConnections(fMasterConnectionList[params->fName]);
  796. }
  797. fMasterList.erase(master_it);
  798. delete (*master_it);
  799. return 1;
  800. }
  801. return 0;
  802. }
  803. }//namespace
  804. static Jack::JackNetMasterManager* master_manager = NULL;
  805. #ifdef __cplusplus
  806. extern "C"
  807. {
  808. #endif
  809. SERVER_EXPORT jack_driver_desc_t* jack_get_descriptor()
  810. {
  811. jack_driver_desc_t * desc;
  812. jack_driver_desc_filler_t filler;
  813. jack_driver_param_value_t value;
  814. desc = jack_driver_descriptor_construct("netmanager", JackDriverNone, "netjack multi-cast master component", &filler);
  815. strcpy(value.str, DEFAULT_MULTICAST_IP);
  816. jack_driver_descriptor_add_parameter(desc, &filler, "multicast-ip", 'a', JackDriverParamString, &value, NULL, "Multicast address", NULL);
  817. value.i = DEFAULT_PORT;
  818. jack_driver_descriptor_add_parameter(desc, &filler, "udp-net-port", 'p', JackDriverParamInt, &value, NULL, "UDP port", NULL);
  819. value.i = false;
  820. jack_driver_descriptor_add_parameter(desc, &filler, "auto-connect", 'c', JackDriverParamBool, &value, NULL, "Auto connect netmaster to system ports", NULL);
  821. value.i = false;
  822. jack_driver_descriptor_add_parameter(desc, &filler, "auto-save", 's', JackDriverParamBool, &value, NULL, "Save/restore netmaster connection state when restarted", NULL);
  823. return desc;
  824. }
  825. SERVER_EXPORT int jack_internal_initialize(jack_client_t* jack_client, const JSList* params)
  826. {
  827. if (master_manager) {
  828. jack_error("Master Manager already loaded");
  829. return 1;
  830. } else {
  831. jack_log("Loading Master Manager");
  832. master_manager = new Jack::JackNetMasterManager(jack_client, params);
  833. return (master_manager) ? 0 : 1;
  834. }
  835. }
  836. SERVER_EXPORT int jack_initialize(jack_client_t* jack_client, const char* load_init)
  837. {
  838. JSList* params = NULL;
  839. bool parse_params = true;
  840. int res = 1;
  841. jack_driver_desc_t* desc = jack_get_descriptor();
  842. Jack::JackArgParser parser(load_init);
  843. if (parser.GetArgc() > 0) {
  844. parse_params = parser.ParseParams(desc, &params);
  845. }
  846. if (parse_params) {
  847. res = jack_internal_initialize(jack_client, params);
  848. parser.FreeParams(params);
  849. }
  850. return res;
  851. }
  852. SERVER_EXPORT void jack_finish(void* arg)
  853. {
  854. if (master_manager) {
  855. jack_log("Unloading Master Manager");
  856. delete master_manager;
  857. master_manager = NULL;
  858. }
  859. }
  860. #ifdef __cplusplus
  861. }
  862. #endif