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.

675 lines
22KB

  1. /*
  2. Copyright (C) 2008 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 "JackError.h"
  17. #include "JackExports.h"
  18. #define DEFAULT_MULTICAST_IP "225.3.19.154"
  19. #define DEFAULT_PORT 19000
  20. using namespace std;
  21. namespace Jack
  22. {
  23. //JackNetMaster******************************************************************************************************
  24. JackNetMaster::JackNetMaster ( JackNetMasterManager* manager, session_params_t& params, struct sockaddr_in& address, struct sockaddr_in& mcast_addr )
  25. {
  26. jack_log ( "JackNetMaster::JackNetMaster" );
  27. //settings
  28. fMasterManager = manager;
  29. fParams = params;
  30. fAddr = address;
  31. fMcastAddr = mcast_addr;
  32. fNSubProcess = fParams.fPeriodSize / fParams.fFramesPerPacket;
  33. fClientName = const_cast<char*> ( fParams.fName );
  34. fNetJumpCnt = 0;
  35. fJackClient = NULL;
  36. fSockfd = 0;
  37. fRunning = false;
  38. //jack audio ports
  39. fAudioCapturePorts = new jack_port_t* [fParams.fSendAudioChannels];
  40. for ( uint port_index = 0; port_index < fParams.fSendAudioChannels; port_index++ )
  41. fAudioCapturePorts[port_index] = NULL;
  42. fAudioPlaybackPorts = new jack_port_t* [fParams.fReturnAudioChannels];
  43. for ( uint port_index = 0; port_index < fParams.fReturnAudioChannels; port_index++ )
  44. fAudioPlaybackPorts[port_index] = NULL;
  45. //jack midi ports
  46. fMidiCapturePorts = new jack_port_t* [fParams.fSendMidiChannels];
  47. for ( uint port_index = 0; port_index < fParams.fSendMidiChannels; port_index++ )
  48. fMidiCapturePorts[port_index] = NULL;
  49. fMidiPlaybackPorts = new jack_port_t* [fParams.fReturnMidiChannels];
  50. for ( uint port_index = 0; port_index < fParams.fReturnMidiChannels; port_index++ )
  51. fMidiPlaybackPorts[port_index] = NULL;
  52. //TX header init
  53. strcpy ( fTxHeader.fPacketType, "header" );
  54. fTxHeader.fDataStream = 's';
  55. fTxHeader.fID = fParams.fID;
  56. fTxHeader.fCycle = 0;
  57. fTxHeader.fSubCycle = 0;
  58. fTxHeader.fMidiDataSize = 0;
  59. fTxHeader.fBitdepth = fParams.fBitdepth;
  60. //RX header init
  61. strcpy ( fRxHeader.fPacketType, "header" );
  62. fRxHeader.fDataStream = 'r';
  63. fRxHeader.fID = fParams.fID;
  64. fRxHeader.fCycle = 0;
  65. fRxHeader.fSubCycle = 0;
  66. fRxHeader.fMidiDataSize = 0;
  67. fRxHeader.fBitdepth = fParams.fBitdepth;
  68. //network buffers
  69. fTxBuffer = new char [fParams.fMtu];
  70. fRxBuffer = new char [fParams.fMtu];
  71. //net audio buffers
  72. fTxData = fTxBuffer + sizeof ( packet_header_t );
  73. fRxData = fRxBuffer + sizeof ( packet_header_t );
  74. //midi net buffers
  75. fNetMidiCaptureBuffer = new NetMidiBuffer ( &fParams, fParams.fSendMidiChannels, fTxData );
  76. fNetMidiPlaybackBuffer = new NetMidiBuffer ( &fParams, fParams.fReturnMidiChannels, fRxData );
  77. //audio net buffers
  78. fNetAudioCaptureBuffer = new NetAudioBuffer ( &fParams, fParams.fSendAudioChannels, fTxData );
  79. fNetAudioPlaybackBuffer = new NetAudioBuffer ( &fParams, fParams.fReturnAudioChannels, fRxData );
  80. //audio netbuffer length
  81. fAudioTxLen = sizeof ( packet_header_t ) + fNetAudioCaptureBuffer->GetSize();
  82. fAudioRxLen = sizeof ( packet_header_t ) + fNetAudioPlaybackBuffer->GetSize();
  83. }
  84. JackNetMaster::~JackNetMaster()
  85. {
  86. jack_log ( "JackNetMaster::~JackNetMaster, ID %u.", fParams.fID );
  87. if ( fJackClient )
  88. {
  89. jack_deactivate ( fJackClient );
  90. FreePorts();
  91. jack_client_close ( fJackClient );
  92. }
  93. if ( fSockfd )
  94. close ( fSockfd );
  95. delete fNetAudioCaptureBuffer;
  96. delete fNetAudioPlaybackBuffer;
  97. delete fNetMidiCaptureBuffer;
  98. delete fNetMidiPlaybackBuffer;
  99. delete[] fAudioCapturePorts;
  100. delete[] fAudioPlaybackPorts;
  101. delete[] fMidiCapturePorts;
  102. delete[] fMidiPlaybackPorts;
  103. delete[] fTxBuffer;
  104. delete[] fRxBuffer;
  105. }
  106. bool JackNetMaster::Init()
  107. {
  108. jack_log ( "JackNetMaster::Init, ID %u.", fParams.fID );
  109. session_params_t params;
  110. struct timeval timeout;
  111. timeout.tv_sec = 1;
  112. timeout.tv_usec = 0;
  113. size_t attempt = 0;
  114. int rx_bytes = 0;
  115. //socket
  116. if ( ( fSockfd = socket ( AF_INET, SOCK_DGRAM, 0 ) ) < 0 )
  117. {
  118. jack_error ( "Can't create socket : %s", strerror ( errno ) );
  119. return false;
  120. }
  121. //timeout on receive (for init)
  122. if ( setsockopt ( fSockfd, SOL_SOCKET, SO_RCVTIMEO, &timeout, sizeof ( timeout ) ) < 0 )
  123. jack_error ( "Can't set timeout : %s", strerror ( errno ) );
  124. //connect
  125. if ( connect ( fSockfd, reinterpret_cast<sockaddr*> ( &fAddr ), sizeof ( struct sockaddr ) ) < 0 )
  126. {
  127. jack_error ( "Can't connect : %s", strerror ( errno ) );
  128. return false;
  129. }
  130. //send 'SLAVE_SETUP' until 'START_MASTER' received
  131. jack_info ( "Sending parameters to %s ...", fParams.fSlaveNetName );
  132. do
  133. {
  134. SetPacketType ( &fParams, SLAVE_SETUP );
  135. if ( send ( fSockfd, &fParams, sizeof ( session_params_t ), 0 ) < 0 )
  136. jack_error ( "Error in send : ", strerror ( errno ) );
  137. if ( ( ( rx_bytes = recv ( fSockfd, &params, sizeof ( session_params_t ), 0 ) ) < 0 ) && ( errno != EAGAIN ) )
  138. {
  139. jack_error ( "Problem with network." );
  140. return false;
  141. }
  142. }
  143. while ( ( GetPacketType ( &params ) != START_MASTER ) && ( ++attempt < 5 ) );
  144. if ( attempt == 5 )
  145. {
  146. jack_error ( "Slave doesn't respond, exiting." );
  147. return false;
  148. }
  149. //set the new timeout for the socket
  150. if ( SetRxTimeout ( &fSockfd, &fParams ) < 0 )
  151. {
  152. jack_error ( "Can't set rx timeout : %s", strerror ( errno ) );
  153. return false;
  154. }
  155. //jack client and process
  156. jack_status_t status;
  157. jack_options_t options = JackNullOption;
  158. if ( ( fJackClient = jack_client_open ( fClientName, options, &status, NULL ) ) == NULL )
  159. {
  160. jack_error ( "Can't open a new jack client." );
  161. return false;
  162. }
  163. jack_set_process_callback ( fJackClient, SetProcess, this );
  164. //port registering
  165. uint i;
  166. char name[24];
  167. //audio
  168. for ( i = 0; i < fParams.fSendAudioChannels; i++ )
  169. {
  170. sprintf ( name, "to_slave_%d", i+1 );
  171. if ( ( fAudioCapturePorts[i] = jack_port_register ( fJackClient, name, JACK_DEFAULT_AUDIO_TYPE, JackPortIsInput, 0 ) ) == NULL )
  172. goto fail;
  173. }
  174. for ( i = 0; i < fParams.fReturnAudioChannels; i++ )
  175. {
  176. sprintf ( name, "from_slave_%d", i+1 );
  177. if ( ( fAudioPlaybackPorts[i] = jack_port_register ( fJackClient, name, JACK_DEFAULT_AUDIO_TYPE, JackPortIsOutput, 0 ) ) == NULL )
  178. goto fail;
  179. }
  180. //midi
  181. for ( i = 0; i < fParams.fSendMidiChannels; i++ )
  182. {
  183. sprintf ( name, "midi_to_slave_%d", i+1 );
  184. if ( ( fMidiCapturePorts[i] = jack_port_register ( fJackClient, name, JACK_DEFAULT_MIDI_TYPE, JackPortIsInput, 0 ) ) == NULL )
  185. goto fail;
  186. }
  187. for ( i = 0; i < fParams.fReturnMidiChannels; i++ )
  188. {
  189. sprintf ( name, "midi_from_slave_%d", i+1 );
  190. if ( ( fMidiPlaybackPorts[i] = jack_port_register ( fJackClient, name, JACK_DEFAULT_MIDI_TYPE, JackPortIsOutput, 0 ) ) == NULL )
  191. goto fail;
  192. }
  193. fRunning = true;
  194. //finally activate jack client
  195. if ( jack_activate ( fJackClient ) != 0 )
  196. {
  197. jack_error ( "Can't activate jack client." );
  198. goto fail;
  199. }
  200. jack_info ( "NetJack new master started." );
  201. return true;
  202. fail:
  203. FreePorts();
  204. jack_client_close ( fJackClient );
  205. fJackClient = NULL;
  206. return false;
  207. }
  208. void JackNetMaster::FreePorts()
  209. {
  210. jack_log ( "JackNetMaster::FreePorts, ID %u", fParams.fID );
  211. uint port_index;
  212. for ( port_index = 0; port_index < fParams.fSendAudioChannels; port_index++ )
  213. if ( fAudioCapturePorts[port_index] )
  214. jack_port_unregister ( fJackClient, fAudioCapturePorts[port_index] );
  215. for ( port_index = 0; port_index < fParams.fReturnAudioChannels; port_index++ )
  216. if ( fAudioPlaybackPorts[port_index] )
  217. jack_port_unregister ( fJackClient, fAudioPlaybackPorts[port_index] );
  218. for ( port_index = 0; port_index < fParams.fSendMidiChannels; port_index++ )
  219. if ( fMidiCapturePorts[port_index] )
  220. jack_port_unregister ( fJackClient, fMidiCapturePorts[port_index] );
  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. void JackNetMaster::Exit()
  226. {
  227. jack_log ( "JackNetMaster::Exit, ID %u", fParams.fID );
  228. //stop process
  229. fRunning = false;
  230. //send a 'multicast euthanasia request' - new socket is required on macosx
  231. jack_info ( "Exiting '%s'", fParams.fName );
  232. SetPacketType ( &fParams, KILL_MASTER );
  233. int mcast_sockfd = socket ( AF_INET, SOCK_DGRAM, 0 );
  234. if ( mcast_sockfd < 0 )
  235. jack_error ( "Can't create socket : %s", strerror ( errno ) );
  236. if ( sendto ( mcast_sockfd, &fParams, sizeof ( session_params_t ), 0,
  237. reinterpret_cast<socket_address_t*> ( &fMcastAddr ), sizeof ( socket_address_t ) ) < 0 )
  238. jack_error ( "Can't send suicide request : %s", strerror ( errno ) );
  239. close ( mcast_sockfd );
  240. }
  241. int JackNetMaster::Send ( char* buffer, size_t size, int flags )
  242. {
  243. int tx_bytes;
  244. if ( ( tx_bytes = send ( fSockfd, buffer, size, flags ) ) < 0 )
  245. {
  246. if ( ( errno == ECONNABORTED ) || ( errno == ECONNREFUSED ) || ( errno == ECONNRESET ) )
  247. {
  248. //fatal connection issue, exit
  249. jack_error ( "'%s' : %s, please check network connection with '%s'.",
  250. fParams.fName, strerror ( errno ), fParams.fSlaveNetName );
  251. Exit();
  252. return 0;
  253. }
  254. else
  255. jack_error ( "Error in send : %s", strerror ( errno ) );
  256. }
  257. return tx_bytes;
  258. }
  259. int JackNetMaster::Recv ( size_t size, int flags )
  260. {
  261. int rx_bytes;
  262. if ( ( rx_bytes = recv ( fSockfd, fRxBuffer, size, flags ) ) < 0 )
  263. {
  264. if ( errno == EAGAIN )
  265. {
  266. //too much receive failure, react...
  267. if ( ++fNetJumpCnt == 100 )
  268. {
  269. jack_error ( "Connection lost, is %s still running ?", fParams.fName );
  270. fNetJumpCnt = 0;
  271. }
  272. return 0;
  273. }
  274. else if ( ( errno == ECONNABORTED ) || ( errno == ECONNREFUSED ) || ( errno == ECONNRESET ) )
  275. {
  276. //fatal connection issue, exit
  277. jack_error ( "'%s' : %s, please check network connection with '%s'.",
  278. fParams.fName, strerror ( errno ), fParams.fSlaveNetName );
  279. Exit();
  280. return 0;
  281. }
  282. else if ( errno != EAGAIN )
  283. jack_error ( "Error in receive : %s", strerror ( errno ) );
  284. }
  285. return rx_bytes;
  286. }
  287. int JackNetMaster::SetProcess ( jack_nframes_t nframes, void* arg )
  288. {
  289. JackNetMaster* master = static_cast<JackNetMaster*> ( arg ); ;
  290. return master->Process();
  291. }
  292. int JackNetMaster::Process()
  293. {
  294. if ( !fRunning )
  295. return 0;
  296. int tx_bytes, rx_bytes, copy_size;
  297. size_t midi_recvd_pckt = 0;
  298. fTxHeader.fCycle++;
  299. fTxHeader.fSubCycle = 0;
  300. fTxHeader.fIsLastPckt = 'n';
  301. packet_header_t* rx_head = reinterpret_cast<packet_header_t*> ( fRxBuffer );
  302. //buffers
  303. uint port_index;
  304. for ( port_index = 0; port_index < fParams.fSendMidiChannels; port_index++ )
  305. fNetMidiCaptureBuffer->fPortBuffer[port_index] =
  306. static_cast<JackMidiBuffer*> ( jack_port_get_buffer ( fMidiCapturePorts[port_index], fParams.fPeriodSize ) );
  307. for ( port_index = 0; port_index < fParams.fSendAudioChannels; port_index++ )
  308. fNetAudioCaptureBuffer->fPortBuffer[port_index] =
  309. static_cast<sample_t*> ( jack_port_get_buffer ( fAudioCapturePorts[port_index], fParams.fPeriodSize ) );
  310. for ( port_index = 0; port_index < fParams.fReturnMidiChannels; port_index++ )
  311. fNetMidiPlaybackBuffer->fPortBuffer[port_index] =
  312. static_cast<JackMidiBuffer*> ( jack_port_get_buffer ( fMidiPlaybackPorts[port_index], fParams.fPeriodSize ) );
  313. for ( port_index = 0; port_index < fParams.fReturnAudioChannels; port_index++ )
  314. fNetAudioPlaybackBuffer->fPortBuffer[port_index] =
  315. static_cast<sample_t*> ( jack_port_get_buffer ( fAudioPlaybackPorts[port_index], fParams.fPeriodSize ) );
  316. //send ------------------------------------------------------------------------------------------------------------------
  317. //sync
  318. fTxHeader.fDataType = 's';
  319. if ( !fParams.fSendMidiChannels && !fParams.fSendAudioChannels )
  320. fTxHeader.fIsLastPckt = 'y';
  321. tx_bytes = Send ( reinterpret_cast<char*> ( &fTxHeader ), sizeof ( packet_header_t ), 0 );
  322. if ( tx_bytes < 1 )
  323. return tx_bytes;
  324. //midi
  325. if ( fParams.fSendMidiChannels )
  326. {
  327. fTxHeader.fDataType = 'm';
  328. fTxHeader.fMidiDataSize = fNetMidiCaptureBuffer->RenderFromJackPorts();
  329. fTxHeader.fNMidiPckt = GetNMidiPckt ( &fParams, fTxHeader.fMidiDataSize );
  330. for ( uint subproc = 0; subproc < fTxHeader.fNMidiPckt; subproc++ )
  331. {
  332. fTxHeader.fSubCycle = subproc;
  333. if ( ( subproc == ( fTxHeader.fNMidiPckt - 1 ) ) && !fParams.fSendAudioChannels )
  334. fTxHeader.fIsLastPckt = 'y';
  335. memcpy ( fTxBuffer, &fTxHeader, sizeof ( packet_header_t ) );
  336. copy_size = fNetMidiCaptureBuffer->RenderToNetwork ( subproc, fTxHeader.fMidiDataSize );
  337. tx_bytes = Send ( fTxBuffer, sizeof ( packet_header_t ) + copy_size, 0 );
  338. if ( tx_bytes < 1 )
  339. return tx_bytes;
  340. }
  341. }
  342. //audio
  343. if ( fParams.fSendAudioChannels )
  344. {
  345. fTxHeader.fDataType = 'a';
  346. for ( uint subproc = 0; subproc < fNSubProcess; subproc++ )
  347. {
  348. fTxHeader.fSubCycle = subproc;
  349. if ( subproc == ( fNSubProcess - 1 ) )
  350. fTxHeader.fIsLastPckt = 'y';
  351. memcpy ( fTxBuffer, &fTxHeader, sizeof ( packet_header_t ) );
  352. fNetAudioCaptureBuffer->RenderFromJackPorts ( subproc );
  353. tx_bytes = Send ( fTxBuffer, fAudioTxLen, 0 );
  354. if ( tx_bytes < 1 )
  355. return tx_bytes;
  356. }
  357. }
  358. //receive ( if there is stg to receive...)-------------------------------------------------------------------------------------
  359. if ( fParams.fReturnMidiChannels || fParams.fReturnAudioChannels )
  360. {
  361. do
  362. {
  363. rx_bytes = Recv ( fParams.fMtu, MSG_PEEK );
  364. if ( rx_bytes < 1 )
  365. return rx_bytes;
  366. if ( rx_bytes && ( rx_head->fDataStream == 'r' ) && ( rx_head->fID == fParams.fID ) )
  367. {
  368. switch ( rx_head->fDataType )
  369. {
  370. case 'm': //midi
  371. rx_bytes = Recv ( rx_bytes, MSG_DONTWAIT );
  372. fRxHeader.fIsLastPckt = rx_head->fIsLastPckt;
  373. fNetMidiPlaybackBuffer->RenderFromNetwork ( rx_head->fSubCycle, rx_bytes - sizeof ( packet_header_t ) );
  374. if ( ++midi_recvd_pckt == rx_head->fNMidiPckt )
  375. fNetMidiPlaybackBuffer->RenderToJackPorts();
  376. fNetJumpCnt = 0;
  377. break;
  378. case 'a': //audio
  379. rx_bytes = Recv ( fAudioRxLen, MSG_DONTWAIT );
  380. if ( !IsNextPacket ( &fRxHeader, rx_head, fNSubProcess ) )
  381. jack_error ( "Packet(s) missing from '%s'...", fParams.fName );
  382. fRxHeader.fCycle = rx_head->fCycle;
  383. fRxHeader.fSubCycle = rx_head->fSubCycle;
  384. fRxHeader.fIsLastPckt = rx_head->fIsLastPckt;
  385. fNetAudioPlaybackBuffer->RenderToJackPorts ( rx_head->fSubCycle );
  386. fNetJumpCnt = 0;
  387. break;
  388. }
  389. }
  390. }
  391. while ( fRxHeader.fIsLastPckt != 'y' );
  392. }
  393. return 0;
  394. }
  395. //JackNetMasterManager***********************************************************************************************
  396. JackNetMasterManager::JackNetMasterManager ( jack_client_t* client )
  397. {
  398. jack_log ( "JackNetMasterManager::JackNetMasterManager" );
  399. fManagerClient = client;
  400. fManagerName = jack_get_client_name ( fManagerClient );
  401. fMCastIP = DEFAULT_MULTICAST_IP;
  402. fUDPPort = DEFAULT_PORT;
  403. fGlobalID = 0;
  404. fRunning = true;
  405. //launch the manager thread
  406. if ( jack_client_create_thread ( fManagerClient, &fManagerThread, 0, 0, NetManagerThread, this ) )
  407. jack_error ( "Can't create the network manager control thread." );
  408. }
  409. JackNetMasterManager::~JackNetMasterManager()
  410. {
  411. jack_log ( "JackNetMasterManager::~JackNetMasterManager" );
  412. Exit();
  413. master_list_t::iterator it;
  414. for ( it = fMasterList.begin(); it != fMasterList.end(); it++ )
  415. delete ( *it );
  416. }
  417. void* JackNetMasterManager::NetManagerThread ( void* arg )
  418. {
  419. jack_info ( "Starting Jack Network Manager." );
  420. JackNetMasterManager* master_manager = static_cast<JackNetMasterManager*> ( arg );
  421. master_manager->Run();
  422. return NULL;
  423. }
  424. void JackNetMasterManager::Run()
  425. {
  426. jack_log ( "JackNetMasterManager::Run" );
  427. //utility variables
  428. socklen_t addr_len = sizeof ( socket_address_t );
  429. char disable = 0;
  430. struct timeval timeout;
  431. timeout.tv_sec = 2;
  432. timeout.tv_usec = 0;
  433. int attempt = 0;
  434. //network
  435. int mcast_sockfd;
  436. struct ip_mreq multicast_req;
  437. struct sockaddr_in mcast_addr;
  438. struct sockaddr_in response_addr;
  439. //data
  440. session_params_t params;
  441. int rx_bytes = 0;
  442. JackNetMaster* net_master;
  443. //socket
  444. if ( ( mcast_sockfd = socket ( AF_INET, SOCK_DGRAM, 0 ) ) < 0 )
  445. {
  446. jack_error ( "Can't create the network management input socket : %s", strerror ( errno ) );
  447. return;
  448. }
  449. //set the multicast address
  450. mcast_addr.sin_family = AF_INET;
  451. mcast_addr.sin_port = htons ( fUDPPort );
  452. if ( inet_aton ( fMCastIP, &mcast_addr.sin_addr ) < 0 )
  453. {
  454. jack_error ( "Cant set multicast address : %s", strerror ( errno ) );
  455. close ( mcast_sockfd );
  456. return;
  457. }
  458. memset ( &mcast_addr.sin_zero, 0, 8 );
  459. //bind the socket to the multicast address
  460. if ( bind ( mcast_sockfd, reinterpret_cast<socket_address_t *> ( &mcast_addr ), addr_len ) < 0 )
  461. {
  462. jack_error ( "Can't bind the network manager socket : %s", strerror ( errno ) );
  463. close ( mcast_sockfd );
  464. return;
  465. }
  466. //join multicast group
  467. inet_aton ( fMCastIP, &multicast_req.imr_multiaddr );
  468. multicast_req.imr_interface.s_addr = htonl ( INADDR_ANY );
  469. if ( setsockopt ( mcast_sockfd, IPPROTO_IP, IP_ADD_MEMBERSHIP, &multicast_req, sizeof ( multicast_req ) ) < 0 )
  470. {
  471. jack_error ( "Can't join multicast group : %s", strerror ( errno ) );
  472. close ( mcast_sockfd );
  473. return;
  474. }
  475. //disable local loop
  476. if ( setsockopt ( mcast_sockfd, IPPROTO_IP, IP_MULTICAST_LOOP, &disable, sizeof ( disable ) ) < 0 )
  477. jack_error ( "Can't set multicast loop option : %s", strerror ( errno ) );
  478. //set a timeout on the multicast receive (the thread can now be cancelled)
  479. if ( setsockopt ( mcast_sockfd, SOL_SOCKET, SO_RCVTIMEO, &timeout, sizeof ( timeout ) ) < 0 )
  480. jack_error ( "Can't set timeout : %s", strerror ( errno ) );
  481. jack_info ( "Waiting for a slave..." );
  482. //main loop, wait for data, deal with it and wait again
  483. do
  484. {
  485. rx_bytes = recvfrom ( mcast_sockfd, &params, sizeof ( session_params_t ), 0,
  486. reinterpret_cast<socket_address_t*> ( &response_addr ), &addr_len );
  487. if ( ( rx_bytes < 0 ) && ( errno != EAGAIN ) )
  488. {
  489. jack_error ( "Error in receive : %s", strerror ( errno ) );
  490. if ( ++attempt == 10 )
  491. {
  492. jack_error ( "Can't receive on the socket, exiting net manager." );
  493. return;
  494. }
  495. }
  496. if ( rx_bytes == sizeof ( session_params_t ) )
  497. {
  498. switch ( GetPacketType ( &params ) )
  499. {
  500. case SLAVE_AVAILABLE:
  501. if ( ( net_master = MasterInit ( params, response_addr, mcast_addr ) ) )
  502. SessionParamsDisplay ( &net_master->fParams );
  503. else
  504. jack_error ( "Can't init new net master..." );
  505. jack_info ( "Waiting for a slave..." );
  506. break;
  507. case KILL_MASTER:
  508. KillMaster ( &params );
  509. jack_info ( "Waiting for a slave..." );
  510. break;
  511. default:
  512. break;
  513. }
  514. }
  515. }
  516. while ( fRunning );
  517. close ( mcast_sockfd );
  518. }
  519. void JackNetMasterManager::Exit()
  520. {
  521. jack_log ( "JackNetMasterManager::Exit" );
  522. fRunning = false;
  523. pthread_join ( fManagerThread, NULL );
  524. jack_info ( "Exiting net manager..." );
  525. }
  526. JackNetMaster* JackNetMasterManager::MasterInit ( session_params_t& params, struct sockaddr_in& address, struct sockaddr_in& mcast_addr )
  527. {
  528. jack_log ( "JackNetMasterManager::MasterInit, Slave : %s", params.fName );
  529. //settings
  530. gethostname ( params.fMasterNetName, 255 );
  531. params.fID = ++fGlobalID;
  532. params.fSampleRate = jack_get_sample_rate ( fManagerClient );
  533. params.fPeriodSize = jack_get_buffer_size ( fManagerClient );
  534. params.fBitdepth = 0;
  535. SetFramesPerPacket ( &params );
  536. SetSlaveName ( params );
  537. //create a new master and add it to the list
  538. JackNetMaster* master = new JackNetMaster ( this, params, address, mcast_addr );
  539. if ( master->Init() )
  540. {
  541. fMasterList.push_back ( master );
  542. return master;
  543. }
  544. delete master;
  545. return NULL;
  546. }
  547. void JackNetMasterManager::SetSlaveName ( session_params_t& params )
  548. {
  549. jack_log ( "JackNetMasterManager::SetSlaveName" );
  550. master_list_it_t it;
  551. for ( it = fMasterList.begin(); it != fMasterList.end(); it++ )
  552. if ( strcmp ( ( *it )->fParams.fName, params.fName ) == 0 )
  553. sprintf ( params.fName, "%s-%u", params.fName, params.fID );
  554. }
  555. master_list_it_t JackNetMasterManager::FindMaster ( uint32_t id )
  556. {
  557. jack_log ( "JackNetMasterManager::FindMaster, ID %u.", id );
  558. master_list_it_t it;
  559. for ( it = fMasterList.begin(); it != fMasterList.end(); it++ )
  560. if ( ( *it )->fParams.fID == id )
  561. return it;
  562. return it;
  563. }
  564. void JackNetMasterManager::KillMaster ( session_params_t* params )
  565. {
  566. jack_log ( "JackNetMasterManager::KillMaster, ID %u.", params->fID );
  567. master_list_it_t master = FindMaster ( params->fID );
  568. if ( master != fMasterList.end() )
  569. {
  570. fMasterList.erase ( master );
  571. delete *master;
  572. }
  573. }
  574. }//namespace
  575. static Jack::JackNetMasterManager* master_manager = NULL;
  576. #ifdef __cplusplus
  577. extern "C"
  578. {
  579. #endif
  580. EXPORT int jack_initialize ( jack_client_t* jack_client, const char* load_init )
  581. {
  582. if ( master_manager )
  583. {
  584. jack_error ( "Master Manager already loaded" );
  585. return 1;
  586. }
  587. else
  588. {
  589. jack_log ( "Loading Master Manager" );
  590. master_manager = new Jack::JackNetMasterManager ( jack_client );
  591. return ( master_manager ) ? 0 : 1;
  592. }
  593. }
  594. EXPORT void jack_finish ( void* arg )
  595. {
  596. if ( master_manager )
  597. {
  598. jack_log ( "Unloading Master Manager" );
  599. delete master_manager;
  600. master_manager = NULL;
  601. }
  602. }
  603. #ifdef __cplusplus
  604. }
  605. #endif