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.

319 lines
10KB

  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. #if defined(HAVE_CONFIG_H)
  16. #include "config.h"
  17. #endif
  18. #include "JackAlsaAdapter.h"
  19. #include "JackServerGlobals.h"
  20. #include "JackEngineControl.h"
  21. namespace Jack
  22. {
  23. JackAlsaAdapter::JackAlsaAdapter ( jack_nframes_t buffer_size, jack_nframes_t sample_rate, const JSList* params ) :
  24. JackAudioAdapterInterface ( buffer_size, sample_rate ),
  25. fThread ( this ),
  26. fAudioInterface ( buffer_size, sample_rate )
  27. {
  28. const JSList* node;
  29. const jack_driver_param_t* param;
  30. fCaptureChannels = 2;
  31. fPlaybackChannels = 2;
  32. fAudioInterface.fPeriod = 2;
  33. for ( node = params; node; node = jack_slist_next ( node ) )
  34. {
  35. param = ( const jack_driver_param_t* ) node->data;
  36. switch ( param->character )
  37. {
  38. case 'i':
  39. fCaptureChannels = param->value.ui;
  40. break;
  41. case 'o':
  42. fPlaybackChannels = param->value.ui;
  43. break;
  44. case 'C':
  45. break;
  46. case 'P':
  47. break;
  48. case 'D':
  49. break;
  50. case 'n':
  51. fAudioInterface.fPeriod = param->value.ui;
  52. break;
  53. case 'd':
  54. fAudioInterface.fCardName = strdup ( param->value.str );
  55. break;
  56. case 'r':
  57. SetAdaptedSampleRate ( param->value.ui );
  58. break;
  59. case 'p':
  60. SetAdaptedBufferSize ( param->value.ui );
  61. break;
  62. case 'q':
  63. fQuality = param->value.ui;
  64. break;
  65. }
  66. }
  67. fAudioInterface.setInputs ( fCaptureChannels );
  68. fAudioInterface.setOutputs ( fPlaybackChannels );
  69. }
  70. int JackAlsaAdapter::Open()
  71. {
  72. //open audio interface
  73. if ( fAudioInterface.open() )
  74. return -1;
  75. //start adapter thread
  76. if ( fThread.StartSync() < 0 )
  77. {
  78. jack_error ( "Cannot start audioadapter thread" );
  79. return -1;
  80. }
  81. //display card info
  82. fAudioInterface.longinfo();
  83. //turn the thread realtime
  84. fThread.AcquireRealTime ( JackServerGlobals::fInstance->GetEngineControl()->fClientPriority );
  85. return 0;
  86. }
  87. int JackAlsaAdapter::Close()
  88. {
  89. #ifdef JACK_MONITOR
  90. fTable.Save();
  91. #endif
  92. switch ( fThread.GetStatus() )
  93. {
  94. // Kill the thread in Init phase
  95. case JackThread::kStarting:
  96. case JackThread::kIniting:
  97. if ( fThread.Kill() < 0 )
  98. {
  99. jack_error ( "Cannot kill thread" );
  100. return -1;
  101. }
  102. break;
  103. // Stop when the thread cycle is finished
  104. case JackThread::kRunning:
  105. if ( fThread.Stop() < 0 )
  106. {
  107. jack_error ( "Cannot stop thread" );
  108. return -1;
  109. }
  110. break;
  111. default:
  112. break;
  113. }
  114. return fAudioInterface.close();
  115. }
  116. bool JackAlsaAdapter::Init()
  117. {
  118. //fill the hardware buffers
  119. for ( unsigned int i = 0; i < fAudioInterface.fPeriod; i++ )
  120. fAudioInterface.write();
  121. return true;
  122. }
  123. bool JackAlsaAdapter::Execute()
  124. {
  125. //read data from audio interface
  126. if ( fAudioInterface.read() < 0 )
  127. return false;
  128. bool failure = false;
  129. //compute resampling factor
  130. jack_nframes_t time1, time2;
  131. ResampleFactor ( time1, time2 );
  132. //resample inputs
  133. for ( int i = 0; i < fCaptureChannels; i++ )
  134. {
  135. fCaptureRingBuffer[i]->SetRatio ( time1, time2 );
  136. if ( fCaptureRingBuffer[i]->WriteResample ( fAudioInterface.fInputSoftChannels[i], fAdaptedBufferSize ) < fAdaptedBufferSize )
  137. failure = true;
  138. }
  139. //resample outputs
  140. for ( int i = 0; i < fPlaybackChannels; i++ )
  141. {
  142. fPlaybackRingBuffer[i]->SetRatio ( time2, time1 );
  143. if ( fPlaybackRingBuffer[i]->ReadResample ( fAudioInterface.fOutputSoftChannels[i], fAdaptedBufferSize ) < fAdaptedBufferSize )
  144. failure = true;
  145. }
  146. #ifdef JACK_MONITOR
  147. fTable.Write ( time1, time2, double ( time1 ) / double ( time2 ), double ( time2 ) / double ( time1 ),
  148. fCaptureRingBuffer[0]->ReadSpace(), fPlaybackRingBuffer[0]->WriteSpace() );
  149. #endif
  150. //write data to audio interface
  151. if ( fAudioInterface.write() < 0 )
  152. return false;
  153. //reset all ringbuffers in case of failure
  154. if ( failure )
  155. {
  156. jack_error ( "JackAlsaAdapter::Execute ringbuffer failure... reset" );
  157. ResetRingBuffers();
  158. }
  159. return true;
  160. }
  161. int JackAlsaAdapter::SetSampleRate ( jack_nframes_t sample_rate )
  162. {
  163. JackAudioAdapterInterface::SetHostSampleRate ( sample_rate );
  164. Close();
  165. return Open();
  166. }
  167. int JackAlsaAdapter::SetBufferSize ( jack_nframes_t buffer_size )
  168. {
  169. JackAudioAdapterInterface::SetHostBufferSize ( buffer_size );
  170. Close();
  171. return Open();
  172. }
  173. } // namespace
  174. #ifdef __cplusplus
  175. extern "C"
  176. {
  177. #endif
  178. SERVER_EXPORT jack_driver_desc_t* jack_get_descriptor()
  179. {
  180. jack_driver_desc_t *desc;
  181. unsigned int i;
  182. desc = ( jack_driver_desc_t* ) calloc ( 1, sizeof ( jack_driver_desc_t ) );
  183. strcpy ( desc->name, "audioadapter" ); // size MUST be less then JACK_DRIVER_NAME_MAX + 1
  184. strcpy ( desc->desc, "netjack audio <==> net backend adapter" ); // size MUST be less then JACK_DRIVER_PARAM_DESC + 1
  185. desc->nparams = 10;
  186. desc->params = ( jack_driver_param_desc_t* ) calloc ( desc->nparams, sizeof ( jack_driver_param_desc_t ) );
  187. i = 0;
  188. strcpy ( desc->params[i].name, "capture" );
  189. desc->params[i].character = 'C';
  190. desc->params[i].type = JackDriverParamString;
  191. strcpy ( desc->params[i].value.str, "none" );
  192. strcpy ( desc->params[i].short_desc,
  193. "Provide capture ports. Optionally set device" );
  194. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  195. i++;
  196. strcpy ( desc->params[i].name, "playback" );
  197. desc->params[i].character = 'P';
  198. desc->params[i].type = JackDriverParamString;
  199. strcpy ( desc->params[i].value.str, "none" );
  200. strcpy ( desc->params[i].short_desc,
  201. "Provide playback ports. Optionally set device" );
  202. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  203. i++;
  204. strcpy ( desc->params[i].name, "device" );
  205. desc->params[i].character = 'd';
  206. desc->params[i].type = JackDriverParamString;
  207. strcpy ( desc->params[i].value.str, "hw:0" );
  208. strcpy ( desc->params[i].short_desc, "ALSA device name" );
  209. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  210. i++;
  211. strcpy ( desc->params[i].name, "rate" );
  212. desc->params[i].character = 'r';
  213. desc->params[i].type = JackDriverParamUInt;
  214. desc->params[i].value.ui = 48000U;
  215. strcpy ( desc->params[i].short_desc, "Sample rate" );
  216. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  217. i++;
  218. strcpy ( desc->params[i].name, "periodsize" );
  219. desc->params[i].character = 'p';
  220. desc->params[i].type = JackDriverParamUInt;
  221. desc->params[i].value.ui = 512U;
  222. strcpy ( desc->params[i].short_desc, "Period size" );
  223. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  224. i++;
  225. strcpy ( desc->params[i].name, "nperiods" );
  226. desc->params[i].character = 'n';
  227. desc->params[i].type = JackDriverParamUInt;
  228. desc->params[i].value.ui = 2U;
  229. strcpy ( desc->params[i].short_desc, "Number of periods of playback latency" );
  230. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  231. i++;
  232. strcpy ( desc->params[i].name, "duplex" );
  233. desc->params[i].character = 'D';
  234. desc->params[i].type = JackDriverParamBool;
  235. desc->params[i].value.i = 1;
  236. strcpy ( desc->params[i].short_desc,
  237. "Provide both capture and playback ports" );
  238. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  239. i++;
  240. strcpy ( desc->params[i].name, "inchannels" );
  241. desc->params[i].character = 'i';
  242. desc->params[i].type = JackDriverParamUInt;
  243. desc->params[i].value.i = 0;
  244. strcpy ( desc->params[i].short_desc,
  245. "Number of capture channels (defaults to hardware max)" );
  246. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  247. i++;
  248. strcpy ( desc->params[i].name, "outchannels" );
  249. desc->params[i].character = 'o';
  250. desc->params[i].type = JackDriverParamUInt;
  251. desc->params[i].value.i = 0;
  252. strcpy ( desc->params[i].short_desc,
  253. "Number of playback channels (defaults to hardware max)" );
  254. strcpy ( desc->params[i].long_desc, desc->params[i].short_desc );
  255. i++;
  256. strcpy(desc->params[i].name, "quality");
  257. desc->params[i].character = 'q';
  258. desc->params[i].type = JackDriverParamInt;
  259. desc->params[i].value.ui = 0;
  260. strcpy(desc->params[i].short_desc, "Resample algorithm quality (0 - 4)");
  261. strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
  262. return desc;
  263. }
  264. #ifdef __cplusplus
  265. }
  266. #endif