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.

452 lines
15KB

  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 "JackNetAdapter.h"
  16. #include "JackException.h"
  17. #include "JackServer.h"
  18. #include "JackEngineControl.h"
  19. #define DEFAULT_MULTICAST_IP "225.3.19.154"
  20. #define DEFAULT_PORT 19000
  21. namespace Jack
  22. {
  23. JackNetAdapter::JackNetAdapter ( jack_nframes_t buffer_size, jack_nframes_t sample_rate, const JSList* params )
  24. : JackAudioAdapterInterface ( buffer_size, sample_rate ), JackNetSlaveInterface(), fThread ( this )
  25. {
  26. jack_log ( "JackNetAdapter::JackNetAdapter" );
  27. if ( SocketAPIInit() < 0 )
  28. jack_error ( "Can't init Socket API, exiting..." );
  29. //global parametering
  30. fMulticastIP = new char[16];
  31. strcpy ( fMulticastIP, DEFAULT_MULTICAST_IP );
  32. uint port = DEFAULT_PORT;
  33. GetHostName ( fParams.fName, JACK_CLIENT_NAME_SIZE );
  34. fSocket.GetName ( fParams.fSlaveNetName );
  35. fParams.fMtu = 1500;
  36. fParams.fTransportSync = 1;
  37. fParams.fSendAudioChannels = 2;
  38. fParams.fReturnAudioChannels = 2;
  39. fParams.fSendMidiChannels = 0;
  40. fParams.fReturnMidiChannels = 0;
  41. fParams.fSampleRate = sample_rate;
  42. fParams.fPeriodSize = buffer_size;
  43. fParams.fSlaveSyncMode = 1;
  44. fParams.fNetworkMode = 'n';
  45. //options parsing
  46. const JSList* node;
  47. const jack_driver_param_t* param;
  48. for ( node = params; node; node = jack_slist_next ( node ) )
  49. {
  50. param = ( const jack_driver_param_t* ) node->data;
  51. switch ( param->character )
  52. {
  53. case 'a' :
  54. if ( strlen ( param->value.str ) < 16 )
  55. strcpy ( fMulticastIP, param->value.str );
  56. else
  57. jack_error ( "Can't use multicast address %s, using default %s", param->value.ui, DEFAULT_MULTICAST_IP );
  58. break;
  59. case 'p' :
  60. fSocket.SetPort ( param->value.ui );
  61. break;
  62. case 'M' :
  63. fParams.fMtu = param->value.i;
  64. break;
  65. case 'C' :
  66. fParams.fSendAudioChannels = param->value.i;
  67. break;
  68. case 'P' :
  69. fParams.fReturnAudioChannels = param->value.i;
  70. break;
  71. case 'n' :
  72. strncpy ( fParams.fName, param->value.str, JACK_CLIENT_NAME_SIZE );
  73. break;
  74. case 't' :
  75. fParams.fTransportSync = param->value.ui;
  76. break;
  77. case 'm' :
  78. if ( strcmp ( param->value.str, "normal" ) == 0 )
  79. fParams.fNetworkMode = 'n';
  80. else if ( strcmp ( param->value.str, "slow" ) == 0 )
  81. fParams.fNetworkMode = 's';
  82. else if ( strcmp ( param->value.str, "fast" ) == 0 )
  83. fParams.fNetworkMode = 'f';
  84. else
  85. jack_error ( "Unknown network mode, using 'normal' mode." );
  86. break;
  87. case 'S' :
  88. fParams.fSlaveSyncMode = 1;
  89. break;
  90. }
  91. }
  92. fSocket.SetPort ( port );
  93. fSocket.SetAddress ( fMulticastIP, port );
  94. SetInputs ( fParams.fSendAudioChannels );
  95. SetOutputs ( fParams.fReturnAudioChannels );
  96. fSoftCaptureBuffer = NULL;
  97. fSoftPlaybackBuffer = NULL;
  98. }
  99. JackNetAdapter::~JackNetAdapter()
  100. {
  101. jack_log ( "JackNetAdapter::~JackNetAdapter" );
  102. int port_index;
  103. if ( fSoftCaptureBuffer )
  104. {
  105. for ( port_index = 0; port_index < fCaptureChannels; port_index++ )
  106. delete[] fSoftCaptureBuffer[port_index];
  107. delete[] fSoftCaptureBuffer;
  108. }
  109. if ( fSoftPlaybackBuffer )
  110. {
  111. for ( port_index = 0; port_index < fPlaybackChannels; port_index++ )
  112. delete[] fSoftPlaybackBuffer[port_index];
  113. delete[] fSoftPlaybackBuffer;
  114. }
  115. }
  116. int JackNetAdapter::Open()
  117. {
  118. jack_log ( "JackNetAdapter::Open" );
  119. jack_info ( "Net adapter started in %s mode %s Master's transport sync.",
  120. ( fParams.fSlaveSyncMode ) ? "sync" : "async", ( fParams.fTransportSync ) ? "with" : "without" );
  121. if ( fThread.StartSync() < 0 )
  122. {
  123. jack_error ( "Cannot start netadapter thread" );
  124. return -1;
  125. }
  126. fThread.AcquireRealTime ( JackServer::fInstance->GetEngineControl()->fPriority - 1 );
  127. return 0;
  128. }
  129. int JackNetAdapter::Close()
  130. {
  131. jack_log ( "JackNetAdapter::Close" );
  132. switch ( fThread.GetStatus() )
  133. {
  134. // Kill the thread in Init phase
  135. case JackThread::kStarting:
  136. case JackThread::kIniting:
  137. if ( fThread.Kill() < 0 )
  138. {
  139. jack_error ( "Cannot kill thread" );
  140. return -1;
  141. }
  142. break;
  143. // Stop when the thread cycle is finished
  144. case JackThread::kRunning:
  145. if ( fThread.Stop() < 0 )
  146. {
  147. jack_error ( "Cannot stop thread" );
  148. return -1;
  149. }
  150. break;
  151. default:
  152. break;
  153. }
  154. fSocket.Close();
  155. return 0;
  156. }
  157. int JackNetAdapter::SetBufferSize ( jack_nframes_t buffer_size )
  158. {
  159. fParams.fPeriodSize = buffer_size;
  160. return 0;
  161. }
  162. bool JackNetAdapter::Init()
  163. {
  164. jack_log ( "JackNetAdapter::Init" );
  165. int port_index;
  166. //init network connection
  167. if ( !JackNetSlaveInterface::Init() )
  168. return false;
  169. //then set global parameters
  170. SetParams();
  171. //set buffers
  172. fSoftCaptureBuffer = new sample_t*[fCaptureChannels];
  173. for ( port_index = 0; port_index < fCaptureChannels; port_index++ )
  174. {
  175. fSoftCaptureBuffer[port_index] = new sample_t[fParams.fPeriodSize];
  176. fNetAudioCaptureBuffer->SetBuffer ( port_index, fSoftCaptureBuffer[port_index] );
  177. }
  178. fSoftPlaybackBuffer = new sample_t*[fPlaybackChannels];
  179. for ( port_index = 0; port_index < fCaptureChannels; port_index++ )
  180. {
  181. fSoftPlaybackBuffer[port_index] = new sample_t[fParams.fPeriodSize];
  182. fNetAudioPlaybackBuffer->SetBuffer ( port_index, fSoftPlaybackBuffer[port_index] );
  183. }
  184. //init done, display parameters
  185. SessionParamsDisplay ( &fParams );
  186. return true;
  187. }
  188. bool JackNetAdapter::Execute()
  189. {
  190. try
  191. {
  192. // Keep running even in case of error
  193. while ( fThread.GetStatus() == JackThread::kRunning )
  194. if ( Process() == SOCKET_ERROR )
  195. return false;
  196. return false;
  197. }
  198. catch ( JackNetException& e )
  199. {
  200. e.PrintMessage();
  201. jack_log ( "NetAdapter is restarted." );
  202. fThread.DropRealTime();
  203. fThread.SetStatus ( JackThread::kIniting );
  204. if ( Init() )
  205. {
  206. fThread.SetStatus ( JackThread::kRunning );
  207. return true;
  208. }
  209. else
  210. return false;
  211. }
  212. }
  213. int JackNetAdapter::Read()
  214. {
  215. if ( SyncRecv() == SOCKET_ERROR )
  216. return 0;
  217. return DataRecv();
  218. }
  219. int JackNetAdapter::Write()
  220. {
  221. if ( SyncSend() == SOCKET_ERROR )
  222. return SOCKET_ERROR;
  223. return DataSend();
  224. }
  225. int JackNetAdapter::Process()
  226. {
  227. bool failure = false;
  228. int port_index;
  229. int rx_bytes, tx_bytes;
  230. //read data from the network
  231. //in case of fatal network error, definitely stop the process
  232. rx_bytes = Read();
  233. if ( rx_bytes == SOCKET_ERROR )
  234. return SOCKET_ERROR;
  235. //if there is data to resample,
  236. if ( rx_bytes )
  237. {
  238. //get the resample factor,
  239. jack_nframes_t time1, time2;
  240. ResampleFactor ( time1, time2 );
  241. //resample input data,
  242. for ( port_index = 0; port_index < fCaptureChannels; port_index++ )
  243. {
  244. fCaptureRingBuffer[port_index]->SetRatio ( time1, time2 );
  245. if ( fCaptureRingBuffer[port_index]->WriteResample ( fSoftCaptureBuffer[port_index], fBufferSize ) < fBufferSize )
  246. failure = true;
  247. }
  248. //and output data,
  249. for ( port_index = 0; port_index < fPlaybackChannels; port_index++ )
  250. {
  251. fPlaybackRingBuffer[port_index]->SetRatio ( time2, time1 );
  252. if ( fPlaybackRingBuffer[port_index]->ReadResample ( fSoftPlaybackBuffer[port_index], fBufferSize ) < fBufferSize )
  253. failure = true;
  254. }
  255. }
  256. //then write data to network
  257. //in case of failure, definitely stop process
  258. if ( Write() == SOCKET_ERROR )
  259. return SOCKET_ERROR;
  260. //if there was any ringbuffer failure during resampling, reset
  261. if ( failure )
  262. {
  263. jack_error ( "JackNetAdapter::Execute ringbuffer failure...reset." );
  264. ResetRingBuffers();
  265. }
  266. return true;
  267. }
  268. } // namespace Jack
  269. #ifdef __cplusplus
  270. extern "C"
  271. {
  272. #endif
  273. #include "driver_interface.h"
  274. #include "JackAudioAdapter.h"
  275. using namespace Jack;
  276. EXPORT jack_driver_desc_t* jack_get_descriptor()
  277. {
  278. jack_driver_desc_t* desc = ( jack_driver_desc_t* ) calloc ( 1, sizeof ( jack_driver_desc_t ) );
  279. strcpy ( desc->name, "net" );
  280. desc->nparams = 9;
  281. desc->params = ( jack_driver_param_desc_t* ) calloc ( desc->nparams, sizeof ( jack_driver_param_desc_t ) );
  282. int i = 0;
  283. strcpy ( desc->params[i].name, "multicast_ip" );
  284. desc->params[i].character = 'a';
  285. desc->params[i].type = JackDriverParamString;
  286. strcpy ( desc->params[i].value.str, DEFAULT_MULTICAST_IP );
  287. strcpy ( desc->params[i].short_desc, "Multicast Address" );
  288. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  289. i++;
  290. strcpy ( desc->params[i].name, "udp_net_port" );
  291. desc->params[i].character = 'p';
  292. desc->params[i].type = JackDriverParamInt;
  293. desc->params[i].value.i = 19000;
  294. strcpy ( desc->params[i].short_desc, "UDP port" );
  295. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  296. i++;
  297. strcpy ( desc->params[i].name, "mtu" );
  298. desc->params[i].character = 'M';
  299. desc->params[i].type = JackDriverParamInt;
  300. desc->params[i].value.i = 1500;
  301. strcpy ( desc->params[i].short_desc, "MTU to the master" );
  302. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  303. i++;
  304. strcpy ( desc->params[i].name, "input_ports" );
  305. desc->params[i].character = 'C';
  306. desc->params[i].type = JackDriverParamInt;
  307. desc->params[i].value.i = 2;
  308. strcpy ( desc->params[i].short_desc, "Number of audio input ports" );
  309. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  310. i++;
  311. strcpy ( desc->params[i].name, "output_ports" );
  312. desc->params[i].character = 'P';
  313. desc->params[i].type = JackDriverParamInt;
  314. desc->params[i].value.i = 2;
  315. strcpy ( desc->params[i].short_desc, "Number of audio output ports" );
  316. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  317. i++;
  318. strcpy ( desc->params[i].name, "client_name" );
  319. desc->params[i].character = 'n';
  320. desc->params[i].type = JackDriverParamString;
  321. strcpy ( desc->params[i].value.str, "'hostname'" );
  322. strcpy ( desc->params[i].short_desc, "Name of the jack client" );
  323. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  324. i++;
  325. strcpy ( desc->params[i].name, "transport_sync" );
  326. desc->params[i].character = 't';
  327. desc->params[i].type = JackDriverParamUInt;
  328. desc->params[i].value.ui = 1U;
  329. strcpy ( desc->params[i].short_desc, "Sync transport with master's" );
  330. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  331. i++;
  332. strcpy ( desc->params[i].name, "mode" );
  333. desc->params[i].character = 'm';
  334. desc->params[i].type = JackDriverParamString;
  335. strcpy ( desc->params[i].value.str, "normal" );
  336. strcpy ( desc->params[i].short_desc, "Slow, Normal or Fast mode." );
  337. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  338. i++;
  339. strcpy ( desc->params[i].name, "sync_mode" );
  340. desc->params[i].character = 'S';
  341. desc->params[i].type = JackDriverParamString;
  342. strcpy ( desc->params[i].value.str, "" );
  343. strcpy ( desc->params[i].short_desc, "Sync mode (same as driver's sync mode) ?" );
  344. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  345. return desc;
  346. }
  347. EXPORT int jack_internal_initialize ( jack_client_t* jack_client, const JSList* params )
  348. {
  349. jack_log ( "Loading netadapter" );
  350. Jack::JackAudioAdapter* adapter;
  351. jack_nframes_t buffer_size = jack_get_buffer_size ( jack_client );
  352. jack_nframes_t sample_rate = jack_get_sample_rate ( jack_client );
  353. adapter = new Jack::JackAudioAdapter ( jack_client, new Jack::JackNetAdapter ( buffer_size, sample_rate, params ) );
  354. assert ( adapter );
  355. if ( adapter->Open() == 0 )
  356. return 0;
  357. else
  358. {
  359. delete adapter;
  360. return 1;
  361. }
  362. }
  363. EXPORT int jack_initialize ( jack_client_t* jack_client, const char* load_init )
  364. {
  365. JSList* params = NULL;
  366. jack_driver_desc_t *desc = jack_get_descriptor();
  367. JackArgParser parser ( load_init );
  368. if ( parser.GetArgc() > 0 )
  369. if ( parser.ParseParams ( desc, &params ) != 0 )
  370. jack_error ( "Internal client : JackArgParser::ParseParams error." );
  371. return jack_internal_initialize ( jack_client, params );
  372. }
  373. EXPORT void jack_finish ( void* arg )
  374. {
  375. Jack::JackAudioAdapter* adapter = static_cast<Jack::JackAudioAdapter*> ( arg );
  376. if ( adapter )
  377. {
  378. jack_log ( "Unloading netadapter" );
  379. adapter->Close();
  380. delete adapter;
  381. }
  382. }
  383. #ifdef __cplusplus
  384. }
  385. #endif