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.

904 lines
32KB

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