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.

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