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.

245 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. #include "JackPortAudioAdapter.h"
  16. #include "JackError.h"
  17. namespace Jack
  18. {
  19. int JackPortAudioAdapter::Render(const void* inputBuffer,
  20. void* outputBuffer,
  21. unsigned long framesPerBuffer,
  22. const PaStreamCallbackTimeInfo* timeInfo,
  23. PaStreamCallbackFlags statusFlags,
  24. void* userData)
  25. {
  26. static_cast<JackPortAudioAdapter*>(userData)->PushAndPull((jack_default_audio_sample_t**)inputBuffer, (jack_default_audio_sample_t**)outputBuffer, framesPerBuffer);
  27. return paContinue;
  28. }
  29. JackPortAudioAdapter::JackPortAudioAdapter(jack_nframes_t buffer_size, jack_nframes_t sample_rate, const JSList* params)
  30. : JackAudioAdapterInterface(buffer_size, sample_rate)
  31. {
  32. jack_log("JackPortAudioAdapter::JackPortAudioAdapter buffer_size = %d, sample_rate = %d", buffer_size, sample_rate);
  33. const JSList* node;
  34. const jack_driver_param_t* param;
  35. int in_max = 0;
  36. int out_max = 0;
  37. fInputDevice = Pa_GetDefaultInputDevice();
  38. fOutputDevice = Pa_GetDefaultOutputDevice();
  39. for (node = params; node; node = jack_slist_next(node)) {
  40. param = (const jack_driver_param_t*) node->data;
  41. switch (param->character)
  42. {
  43. case 'i' :
  44. fCaptureChannels = param->value.ui;
  45. break;
  46. case 'o' :
  47. fPlaybackChannels = param->value.ui;
  48. break;
  49. case 'C' :
  50. if (fPaDevices.GetInputDeviceFromName(param->value.str, fInputDevice, in_max) < 0) {
  51. jack_error("Can't use %s, taking default input device", param->value.str);
  52. fInputDevice = Pa_GetDefaultInputDevice();
  53. }
  54. break;
  55. case 'P' :
  56. if (fPaDevices.GetOutputDeviceFromName(param->value.str, fOutputDevice, out_max) < 0) {
  57. jack_error("Can't use %s, taking default output device", param->value.str);
  58. fOutputDevice = Pa_GetDefaultOutputDevice();
  59. }
  60. break;
  61. case 'r' :
  62. SetAdaptedSampleRate(param->value.ui);
  63. break;
  64. case 'p' :
  65. SetAdaptedBufferSize(param->value.ui);
  66. break;
  67. case 'd' :
  68. if (fPaDevices.GetInputDeviceFromName(param->value.str, fInputDevice, in_max) < 0)
  69. jack_error("Can't use %s, taking default input device", param->value.str);
  70. if (fPaDevices.GetOutputDeviceFromName(param->value.str, fOutputDevice, out_max) < 0)
  71. jack_error("Can't use %s, taking default output device", param->value.str);
  72. break;
  73. case 'l' :
  74. fPaDevices.DisplayDevicesNames();
  75. break;
  76. case 'q':
  77. fQuality = param->value.ui;
  78. break;
  79. case 'g':
  80. fRingbufferCurSize = param->value.ui;
  81. fAdaptative = false;
  82. break;
  83. }
  84. }
  85. //max channels
  86. if (in_max == 0 && fInputDevice != paNoDevice)
  87. in_max = fPaDevices.GetDeviceInfo(fInputDevice)->maxInputChannels;
  88. if (out_max == 0 && fOutputDevice != paNoDevice)
  89. out_max = fPaDevices.GetDeviceInfo(fOutputDevice)->maxOutputChannels;
  90. //effective channels
  91. if ((fCaptureChannels == 0) || (fCaptureChannels > in_max))
  92. fCaptureChannels = in_max;
  93. if ((fPlaybackChannels == 0) || (fPlaybackChannels > out_max))
  94. fPlaybackChannels = out_max;
  95. //set adapter interface channels
  96. SetInputs(fCaptureChannels);
  97. SetOutputs(fPlaybackChannels);
  98. }
  99. int JackPortAudioAdapter::Open()
  100. {
  101. PaError err;
  102. PaStreamParameters inputParameters;
  103. PaStreamParameters outputParameters;
  104. if (fInputDevice == paNoDevice && fOutputDevice == paNoDevice) {
  105. jack_error("No input and output device!!");
  106. return -1;
  107. }
  108. jack_log("JackPortAudioAdapter::Open fInputDevice = %d DeviceName %s", fInputDevice, fPaDevices.GetFullName(fInputDevice).c_str());
  109. jack_log("JackPortAudioAdapter::Open fOutputDevice = %d DeviceName %s", fOutputDevice, fPaDevices.GetFullName(fOutputDevice).c_str());
  110. jack_log("JackPortAudioAdapter::Open fAdaptedBufferSize = %u fAdaptedSampleRate %u", fAdaptedBufferSize, fAdaptedSampleRate);
  111. inputParameters.device = fInputDevice;
  112. inputParameters.channelCount = fCaptureChannels;
  113. inputParameters.sampleFormat = paFloat32 | paNonInterleaved; // 32 bit floating point output
  114. inputParameters.suggestedLatency = (fInputDevice != paNoDevice) // TODO: check how to setup this on ASIO
  115. ? fPaDevices.GetDeviceInfo(fInputDevice)->defaultLowInputLatency
  116. : 0;
  117. inputParameters.hostApiSpecificStreamInfo = NULL;
  118. outputParameters.device = fOutputDevice;
  119. outputParameters.channelCount = fPlaybackChannels;
  120. outputParameters.sampleFormat = paFloat32 | paNonInterleaved; // 32 bit floating point output
  121. outputParameters.suggestedLatency = (fOutputDevice != paNoDevice) // TODO: check how to setup this on ASIO
  122. ? fPaDevices.GetDeviceInfo(fOutputDevice)->defaultLowOutputLatency
  123. : 0;
  124. outputParameters.hostApiSpecificStreamInfo = NULL;
  125. err = Pa_OpenStream( &fStream,
  126. (fInputDevice == paNoDevice) ? 0 : &inputParameters,
  127. (fOutputDevice == paNoDevice) ? 0 : &outputParameters,
  128. fAdaptedSampleRate,
  129. fAdaptedBufferSize,
  130. paNoFlag, // Clipping is on...
  131. Render,
  132. this);
  133. if (err != paNoError) {
  134. jack_error("Pa_OpenStream error = %s", Pa_GetErrorText(err));
  135. return -1;
  136. }
  137. err = Pa_StartStream(fStream);
  138. if (err != paNoError) {
  139. jack_error("Pa_StartStream error = %s", Pa_GetErrorText(err));
  140. return -1;
  141. }
  142. jack_log("JackPortAudioAdapter::Open OK");
  143. return 0;
  144. }
  145. int JackPortAudioAdapter::Close()
  146. {
  147. #ifdef JACK_MONITOR
  148. fTable.Save(fHostBufferSize, fHostSampleRate, fAdaptedSampleRate, fAdaptedBufferSize);
  149. #endif
  150. jack_log("JackPortAudioAdapter::Close");
  151. Pa_StopStream(fStream);
  152. jack_log("JackPortAudioAdapter:: Pa_StopStream");
  153. Pa_CloseStream(fStream);
  154. jack_log("JackPortAudioAdapter:: Pa_CloseStream");
  155. return 0;
  156. }
  157. int JackPortAudioAdapter::SetSampleRate(jack_nframes_t sample_rate)
  158. {
  159. JackAudioAdapterInterface::SetHostSampleRate(sample_rate);
  160. Close();
  161. return Open();
  162. }
  163. int JackPortAudioAdapter::SetBufferSize(jack_nframes_t buffer_size)
  164. {
  165. JackAudioAdapterInterface::SetHostBufferSize(buffer_size);
  166. Close();
  167. return Open();
  168. }
  169. } // namespace
  170. #ifdef __cplusplus
  171. extern "C"
  172. {
  173. #endif
  174. SERVER_EXPORT jack_driver_desc_t* jack_get_descriptor()
  175. {
  176. jack_driver_desc_t * desc;
  177. jack_driver_desc_filler_t filler;
  178. jack_driver_param_value_t value;
  179. desc = jack_driver_descriptor_construct("audioadapter", JackDriverNone, "netjack audio <==> net backend adapter", &filler);
  180. value.ui = 0;
  181. jack_driver_descriptor_add_parameter(desc, &filler, "in-channels", 'i', JackDriverParamUInt, &value, NULL, "Maximum number of input channels", NULL);
  182. jack_driver_descriptor_add_parameter(desc, &filler, "out-channels", 'o', JackDriverParamUInt, &value, NULL, "Maximum number of output channels", NULL);
  183. jack_driver_descriptor_add_parameter(desc, &filler, "capture", 'C', JackDriverParamString, &value, NULL, "Provide capture ports. Optionally set PortAudio device name", NULL);
  184. jack_driver_descriptor_add_parameter(desc, &filler, "playback", 'P', JackDriverParamString, &value, NULL, "Provide playback ports. Optionally set PortAudio device name", NULL);
  185. value.ui = 44100U;
  186. jack_driver_descriptor_add_parameter(desc, &filler, "rate", 'r', JackDriverParamUInt, &value, NULL, "Sample rate", NULL);
  187. value.ui = 512U;
  188. jack_driver_descriptor_add_parameter(desc, &filler, "periodsize", 'p', JackDriverParamUInt, &value, NULL, "Period size", NULL);
  189. jack_driver_descriptor_add_parameter(desc, &filler, "device", 'd', JackDriverParamString, &value, NULL, "PortAudio device name", NULL);
  190. value.i = true;
  191. jack_driver_descriptor_add_parameter(desc, &filler, "list-devices", 'l', JackDriverParamBool, &value, NULL, "Display available PortAudio devices", NULL);
  192. value.ui = 0;
  193. jack_driver_descriptor_add_parameter(desc, &filler, "quality", 'q', JackDriverParamUInt, &value, NULL, "Resample algorithm quality (0 - 4)", NULL);
  194. value.ui = 32768;
  195. jack_driver_descriptor_add_parameter(desc, &filler, "ring-buffer", 'g', JackDriverParamUInt, &value, NULL, "Fixed ringbuffer size", "Fixed ringbuffer size (if not set => automatic adaptative)");
  196. return desc;
  197. }
  198. #ifdef __cplusplus
  199. }
  200. #endif