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.

674 lines
22KB

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