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.

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