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.

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