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.

269 lines
7.8KB

  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 "JackServer.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), fAudioInterface(GetInputs(), GetOutputs(), buffer_size, sample_rate)
  26. {
  27. const JSList* node;
  28. const jack_driver_param_t* param;
  29. fCaptureChannels = 2;
  30. fPlaybackChannels = 2;
  31. for (node = params; node; node = jack_slist_next(node)) {
  32. param = (const jack_driver_param_t*) node->data;
  33. switch (param->character) {
  34. case 'i':
  35. fCaptureChannels = param->value.ui;
  36. break;
  37. case 'o':
  38. fPlaybackChannels = param->value.ui;
  39. break;
  40. case 'C':
  41. break;
  42. case 'P':
  43. break;
  44. case 'D':
  45. break;
  46. case 'n':
  47. fAudioInterface.fPeriod = param->value.ui;
  48. break;
  49. case 'd':
  50. fAudioInterface.fCardName = strdup(param->value.str);
  51. break;
  52. case 'r':
  53. SetAudioSampleRate(param->value.ui);
  54. break;
  55. }
  56. }
  57. }
  58. int JackAlsaAdapter::Open()
  59. {
  60. if (fAudioInterface.open() != 0)
  61. return -1;
  62. if (fThread.StartSync() < 0) {
  63. jack_error("Cannot start audioadapter thread");
  64. return -1;
  65. }
  66. fAudioInterface.longinfo();
  67. fThread.AcquireRealTime(JackServer::fInstance->GetEngineControl()->fPriority);
  68. return 0;
  69. }
  70. int JackAlsaAdapter::Close()
  71. {
  72. #ifdef JACK_MONITOR
  73. fTable.Save();
  74. #endif
  75. switch (fThread.GetStatus()) {
  76. // Kill the thread in Init phase
  77. case JackThread::kStarting:
  78. case JackThread::kIniting:
  79. if (fThread.Kill() < 0) {
  80. jack_error("Cannot kill thread");
  81. return -1;
  82. }
  83. break;
  84. // Stop when the thread cycle is finished
  85. case JackThread::kRunning:
  86. if (fThread.Stop() < 0) {
  87. jack_error("Cannot stop thread");
  88. return -1;
  89. }
  90. break;
  91. default:
  92. break;
  93. }
  94. return fAudioInterface.close();
  95. }
  96. bool JackAlsaAdapter::Init()
  97. {
  98. fAudioInterface.write();
  99. fAudioInterface.write();
  100. return true;
  101. }
  102. bool JackAlsaAdapter::Execute()
  103. {
  104. if (fAudioInterface.read() < 0)
  105. return false;
  106. bool failure = false;
  107. jack_nframes_t time1, time2;
  108. ResampleFactor(time1, time2);
  109. for (int i = 0; i < fCaptureChannels; i++) {
  110. fCaptureRingBuffer[i]->SetRatio(time1, time2);
  111. if (fCaptureRingBuffer[i]->WriteResample(fAudioInterface.fInputSoftChannels[i], fBufferSize) < fBufferSize)
  112. failure = true;
  113. }
  114. for (int i = 0; i < fPlaybackChannels; i++) {
  115. fPlaybackRingBuffer[i]->SetRatio(time2, time1);
  116. if (fPlaybackRingBuffer[i]->ReadResample(fAudioInterface.fOutputSoftChannels[i], fBufferSize) < fBufferSize)
  117. failure = true;
  118. }
  119. #ifdef JACK_MONITOR
  120. fTable.Write(time1, time2, double(time1) / double(time2), double(time2) / double(time1),
  121. fCaptureRingBuffer[0]->ReadSpace(), fPlaybackRingBuffer[0]->WriteSpace());
  122. #endif
  123. if (fAudioInterface.write() < 0)
  124. return false;
  125. // Reset all ringbuffers in case of failure
  126. if (failure) {
  127. jack_error("JackAlsaAdapter::Execute ringbuffer failure... reset");
  128. ResetRingBuffers();
  129. }
  130. return true;
  131. }
  132. int JackAlsaAdapter::SetBufferSize(jack_nframes_t buffer_size)
  133. {
  134. JackAudioAdapterInterface::SetBufferSize(buffer_size);
  135. Close();
  136. return Open();
  137. }
  138. } // namespace
  139. #ifdef __cplusplus
  140. extern "C"
  141. {
  142. #endif
  143. EXPORT jack_driver_desc_t* jack_get_descriptor()
  144. {
  145. jack_driver_desc_t *desc;
  146. jack_driver_param_desc_t * params;
  147. unsigned int i;
  148. desc = (jack_driver_desc_t*)calloc(1, sizeof(jack_driver_desc_t));
  149. strcpy (desc->name, "alsa-adapter");
  150. desc->nparams = 8;
  151. params = (jack_driver_param_desc_t*)calloc(desc->nparams, sizeof(jack_driver_param_desc_t));
  152. i = 0;
  153. strcpy(params[i].name, "capture");
  154. params[i].character = 'C';
  155. params[i].type = JackDriverParamString;
  156. strcpy (params[i].value.str, "none");
  157. strcpy (params[i].short_desc,
  158. "Provide capture ports. Optionally set device");
  159. strcpy (params[i].long_desc, params[i].short_desc);
  160. i++;
  161. strcpy(params[i].name, "playback");
  162. params[i].character = 'P';
  163. params[i].type = JackDriverParamString;
  164. strcpy(params[i].value.str, "none");
  165. strcpy(params[i].short_desc,
  166. "Provide playback ports. Optionally set device");
  167. strcpy(params[i].long_desc, params[i].short_desc);
  168. i++;
  169. strcpy(params[i].name, "device");
  170. params[i].character = 'd';
  171. params[i].type = JackDriverParamString;
  172. strcpy(params[i].value.str, "hw:0");
  173. strcpy(params[i].short_desc, "ALSA device name");
  174. strcpy(params[i].long_desc, params[i].short_desc);
  175. i++;
  176. strcpy (params[i].name, "rate");
  177. params[i].character = 'r';
  178. params[i].type = JackDriverParamUInt;
  179. params[i].value.ui = 48000U;
  180. strcpy(params[i].short_desc, "Sample rate");
  181. strcpy(params[i].long_desc, params[i].short_desc);
  182. i++;
  183. strcpy(params[i].name, "nperiods");
  184. params[i].character = 'n';
  185. params[i].type = JackDriverParamUInt;
  186. params[i].value.ui = 2U;
  187. strcpy(params[i].short_desc, "Number of periods of playback latency");
  188. strcpy(params[i].long_desc, params[i].short_desc);
  189. i++;
  190. strcpy(params[i].name, "duplex");
  191. params[i].character = 'D';
  192. params[i].type = JackDriverParamBool;
  193. params[i].value.i = 1;
  194. strcpy(params[i].short_desc,
  195. "Provide both capture and playback ports");
  196. strcpy(params[i].long_desc, params[i].short_desc);
  197. i++;
  198. strcpy(params[i].name, "inchannels");
  199. params[i].character = 'i';
  200. params[i].type = JackDriverParamUInt;
  201. params[i].value.i = 0;
  202. strcpy(params[i].short_desc,
  203. "Number of capture channels (defaults to hardware max)");
  204. strcpy(params[i].long_desc, params[i].short_desc);
  205. i++;
  206. strcpy(params[i].name, "outchannels");
  207. params[i].character = 'o';
  208. params[i].type = JackDriverParamUInt;
  209. params[i].value.i = 0;
  210. strcpy(params[i].short_desc,
  211. "Number of playback channels (defaults to hardware max)");
  212. strcpy(params[i].long_desc, params[i].short_desc);
  213. desc->params = params;
  214. return desc;
  215. }
  216. #ifdef __cplusplus
  217. }
  218. #endif