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.

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