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.

801 lines
30KB

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