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.

311 lines
12KB

  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 "JackPortAudioAdapter.h"
  19. #include "JackError.h"
  20. namespace Jack
  21. {
  22. int JackPortAudioAdapter::Render(const void* inputBuffer,
  23. void* outputBuffer,
  24. unsigned long framesPerBuffer,
  25. const PaStreamCallbackTimeInfo* timeInfo,
  26. PaStreamCallbackFlags statusFlags,
  27. void* userData)
  28. {
  29. JackPortAudioAdapter* adapter = static_cast<JackPortAudioAdapter*>(userData);
  30. adapter->PushAndPull((float**)inputBuffer, (float**)outputBuffer, framesPerBuffer);
  31. return paContinue;
  32. }
  33. JackPortAudioAdapter::JackPortAudioAdapter ( jack_nframes_t buffer_size, jack_nframes_t sample_rate, const JSList* params )
  34. : JackAudioAdapterInterface ( buffer_size, sample_rate )
  35. {
  36. jack_log ( "JackPortAudioAdapter::JackPortAudioAdapter buffer_size = %d, sample_rate = %d", buffer_size, sample_rate );
  37. const JSList* node;
  38. const jack_driver_param_t* param;
  39. int in_max = 0;
  40. int out_max = 0;
  41. fInputDevice = Pa_GetDefaultInputDevice();
  42. fOutputDevice = Pa_GetDefaultOutputDevice();
  43. for (node = params; node; node = jack_slist_next(node))
  44. {
  45. param = (const jack_driver_param_t*) node->data;
  46. switch (param->character)
  47. {
  48. case 'i' :
  49. fCaptureChannels = param->value.ui;
  50. break;
  51. case 'o' :
  52. fPlaybackChannels = param->value.ui;
  53. break;
  54. case 'C' :
  55. if ( fPaDevices.GetInputDeviceFromName(param->value.str, fInputDevice, in_max) < 0 )
  56. {
  57. jack_error ( "Can't use %s, taking default input device", param->value.str );
  58. fInputDevice = Pa_GetDefaultInputDevice();
  59. }
  60. break;
  61. case 'P' :
  62. if ( fPaDevices.GetOutputDeviceFromName(param->value.str, fOutputDevice, out_max) < 0 )
  63. {
  64. jack_error ( "Can't use %s, taking default output device", param->value.str );
  65. fOutputDevice = Pa_GetDefaultOutputDevice();
  66. }
  67. break;
  68. case 'r' :
  69. SetAdaptedSampleRate ( param->value.ui );
  70. break;
  71. case 'p' :
  72. SetAdaptedBufferSize ( param->value.ui );
  73. break;
  74. case 'd' :
  75. if ( fPaDevices.GetInputDeviceFromName ( param->value.str, fInputDevice, in_max ) < 0 )
  76. jack_error ( "Can't use %s, taking default input device", param->value.str );
  77. if ( fPaDevices.GetOutputDeviceFromName ( param->value.str, fOutputDevice, out_max ) < 0 )
  78. jack_error ( "Can't use %s, taking default output device", param->value.str );
  79. break;
  80. case 'l' :
  81. fPaDevices.DisplayDevicesNames();
  82. break;
  83. case 'q':
  84. fQuality = param->value.ui;
  85. break;
  86. case 'g':
  87. fRingbufferSize = param->value.ui;
  88. break;
  89. }
  90. }
  91. //max channels
  92. if ( in_max == 0 )
  93. in_max = fPaDevices.GetDeviceInfo ( fInputDevice )->maxInputChannels;
  94. if ( out_max == 0 )
  95. out_max = fPaDevices.GetDeviceInfo ( fOutputDevice )->maxOutputChannels;
  96. //effective channels
  97. if ( ( fCaptureChannels == 0 ) || ( fCaptureChannels > in_max ) )
  98. fCaptureChannels = in_max;
  99. if ( ( fPlaybackChannels == 0 ) || ( fPlaybackChannels > out_max ) )
  100. fPlaybackChannels = out_max;
  101. //set adapter interface channels
  102. SetInputs ( fCaptureChannels );
  103. SetOutputs ( fPlaybackChannels );
  104. }
  105. int JackPortAudioAdapter::Open()
  106. {
  107. PaError err;
  108. PaStreamParameters inputParameters;
  109. PaStreamParameters outputParameters;
  110. if ( JackAudioAdapterInterface::Open() < 0 )
  111. return -1;
  112. jack_log("JackPortAudioAdapter::Open fInputDevice = %d DeviceName %s", fInputDevice, fPaDevices.GetFullName(fInputDevice).c_str());
  113. jack_log("JackPortAudioAdapter::Open fOutputDevice = %d DeviceName %s", fOutputDevice, fPaDevices.GetFullName(fOutputDevice).c_str());
  114. jack_log("JackPortAudioAdapter::Open fAdaptedBufferSize = %u fAdaptedSampleRate %u", fAdaptedBufferSize, fAdaptedSampleRate);
  115. inputParameters.device = fInputDevice;
  116. inputParameters.channelCount = fCaptureChannels;
  117. inputParameters.sampleFormat = paFloat32 | paNonInterleaved; // 32 bit floating point output
  118. inputParameters.suggestedLatency = ( fInputDevice != paNoDevice ) // TODO: check how to setup this on ASIO
  119. ? fPaDevices.GetDeviceInfo(fInputDevice)->defaultLowInputLatency
  120. : 0;
  121. inputParameters.hostApiSpecificStreamInfo = NULL;
  122. outputParameters.device = fOutputDevice;
  123. outputParameters.channelCount = fPlaybackChannels;
  124. outputParameters.sampleFormat = paFloat32 | paNonInterleaved; // 32 bit floating point output
  125. outputParameters.suggestedLatency = ( fOutputDevice != paNoDevice ) // TODO: check how to setup this on ASIO
  126. ? fPaDevices.GetDeviceInfo(fOutputDevice)->defaultLowOutputLatency
  127. : 0;
  128. outputParameters.hostApiSpecificStreamInfo = NULL;
  129. err = Pa_OpenStream( &fStream,
  130. ( fInputDevice == paNoDevice ) ? 0 : &inputParameters,
  131. ( fOutputDevice == paNoDevice ) ? 0 : &outputParameters,
  132. fAdaptedSampleRate,
  133. fAdaptedBufferSize,
  134. paNoFlag, // Clipping is on...
  135. Render,
  136. this );
  137. if ( err != paNoError )
  138. {
  139. jack_error ( "Pa_OpenStream error = %s", Pa_GetErrorText ( err ) );
  140. return -1;
  141. }
  142. err = Pa_StartStream ( fStream );
  143. if ( err != paNoError )
  144. {
  145. jack_error ( "Pa_StartStream error = %s", Pa_GetErrorText ( err ) );
  146. return -1;
  147. }
  148. jack_log ( "JackPortAudioAdapter::Open OK" );
  149. return 0;
  150. }
  151. int JackPortAudioAdapter::Close()
  152. {
  153. #ifdef JACK_MONITOR
  154. fTable.Save(fHostBufferSize, fHostSampleRate, fAdaptedSampleRate, fAdaptedBufferSize);
  155. #endif
  156. jack_log ( "JackPortAudioAdapter::Close" );
  157. Pa_StopStream ( fStream );
  158. jack_log ( "JackPortAudioAdapter:: Pa_StopStream" );
  159. Pa_CloseStream ( fStream );
  160. jack_log ( "JackPortAudioAdapter:: Pa_CloseStream" );
  161. return JackAudioAdapterInterface::Close();
  162. }
  163. int JackPortAudioAdapter::SetSampleRate ( jack_nframes_t sample_rate )
  164. {
  165. JackAudioAdapterInterface::SetHostSampleRate ( sample_rate );
  166. Close();
  167. return Open();
  168. }
  169. int JackPortAudioAdapter::SetBufferSize ( jack_nframes_t buffer_size )
  170. {
  171. JackAudioAdapterInterface::SetHostBufferSize ( buffer_size );
  172. Close();
  173. return Open();
  174. }
  175. } // namespace
  176. #ifdef __cplusplus
  177. extern "C"
  178. {
  179. #endif
  180. SERVER_EXPORT jack_driver_desc_t* jack_get_descriptor()
  181. {
  182. jack_driver_desc_t *desc;
  183. unsigned int i;
  184. desc = (jack_driver_desc_t*)calloc(1, sizeof(jack_driver_desc_t));
  185. strcpy(desc->name, "audioadapter"); // size MUST be less then JACK_DRIVER_NAME_MAX + 1
  186. strcpy(desc->desc, "netjack audio <==> net backend adapter"); // size MUST be less then JACK_DRIVER_PARAM_DESC + 1
  187. desc->nparams = 10;
  188. desc->params = (jack_driver_param_desc_t*)calloc(desc->nparams, sizeof(jack_driver_param_desc_t));
  189. i = 0;
  190. strcpy(desc->params[i].name, "inchannels");
  191. desc->params[i].character = 'i';
  192. desc->params[i].type = JackDriverParamInt;
  193. desc->params[i].value.ui = 0;
  194. strcpy(desc->params[i].short_desc, "Maximum number of input channels");
  195. strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
  196. i++;
  197. strcpy(desc->params[i].name, "outchannels");
  198. desc->params[i].character = 'o';
  199. desc->params[i].type = JackDriverParamInt;
  200. desc->params[i].value.ui = 0;
  201. strcpy(desc->params[i].short_desc, "Maximum number of output channels");
  202. strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
  203. i++;
  204. strcpy(desc->params[i].name, "capture");
  205. desc->params[i].character = 'C';
  206. desc->params[i].type = JackDriverParamString;
  207. strcpy(desc->params[i].value.str, "default input device");
  208. strcpy(desc->params[i].short_desc, "Provide capture ports. Optionally set PortAudio device name");
  209. strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
  210. i++;
  211. strcpy(desc->params[i].name, "playback");
  212. desc->params[i].character = 'P';
  213. desc->params[i].type = JackDriverParamString;
  214. strcpy(desc->params[i].value.str, "default output device");
  215. strcpy(desc->params[i].short_desc, "Provide playback ports. Optionally set PortAudio device name");
  216. strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
  217. i++;
  218. strcpy(desc->params[i].name, "rate");
  219. desc->params[i].character = 'r';
  220. desc->params[i].type = JackDriverParamUInt;
  221. desc->params[i].value.ui = 44100U;
  222. strcpy(desc->params[i].short_desc, "Sample rate");
  223. strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
  224. i++;
  225. strcpy(desc->params[i].name, "periodsize");
  226. desc->params[i].character = 'p';
  227. desc->params[i].type = JackDriverParamUInt;
  228. desc->params[i].value.ui = 512U;
  229. strcpy(desc->params[i].short_desc, "Period size");
  230. strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
  231. i++;
  232. strcpy(desc->params[i].name, "device");
  233. desc->params[i].character = 'd';
  234. desc->params[i].type = JackDriverParamString;
  235. desc->params[i].value.ui = 128U;
  236. strcpy(desc->params[i].value.str, "default device");
  237. strcpy(desc->params[i].short_desc, "PortAudio device name");
  238. strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
  239. i++;
  240. strcpy(desc->params[i].name, "list-devices");
  241. desc->params[i].character = 'l';
  242. desc->params[i].type = JackDriverParamBool;
  243. desc->params[i].value.i = TRUE;
  244. strcpy(desc->params[i].short_desc, "Display available PortAudio devices");
  245. strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
  246. i++;
  247. strcpy(desc->params[i].name, "quality");
  248. desc->params[i].character = 'q';
  249. desc->params[i].type = JackDriverParamInt;
  250. desc->params[i].value.ui = 0;
  251. strcpy(desc->params[i].short_desc, "Resample algorithm quality (0 - 4)");
  252. strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
  253. i++;
  254. strcpy(desc->params[i].name, "ring-buffer");
  255. desc->params[i].character = 'g';
  256. desc->params[i].type = JackDriverParamInt;
  257. desc->params[i].value.ui = 0;
  258. strcpy(desc->params[i].short_desc, "Resampling ringbuffer size in frames (default = 16384)");
  259. strcpy(desc->params[i].long_desc, desc->params[i].short_desc);
  260. return desc;
  261. }
  262. #ifdef __cplusplus
  263. }
  264. #endif