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.

906 lines
33KB

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