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.

144 lines
4.7KB

  1. /*
  2. Copyright (C) 2011 Devin Anderson
  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 <memory>
  16. #include "JackALSARawMidiInputPort.h"
  17. #include "JackMidiUtil.h"
  18. using Jack::JackALSARawMidiInputPort;
  19. JackALSARawMidiInputPort::JackALSARawMidiInputPort(snd_rawmidi_info_t *info,
  20. size_t index,
  21. size_t max_bytes,
  22. size_t max_messages):
  23. JackALSARawMidiPort(info, index)
  24. {
  25. alsa_event = 0;
  26. jack_event = 0;
  27. receive_queue = new JackALSARawMidiReceiveQueue(rawmidi, max_bytes);
  28. std::auto_ptr<JackALSARawMidiReceiveQueue> receive_ptr(receive_queue);
  29. thread_queue = new JackMidiAsyncQueue(max_bytes, max_messages);
  30. std::auto_ptr<JackMidiAsyncQueue> thread_ptr(thread_queue);
  31. write_queue = new JackMidiBufferWriteQueue();
  32. std::auto_ptr<JackMidiBufferWriteQueue> write_ptr(write_queue);
  33. raw_queue = new JackMidiRawInputWriteQueue(thread_queue, max_bytes,
  34. max_messages);
  35. write_ptr.release();
  36. thread_ptr.release();
  37. receive_ptr.release();
  38. }
  39. JackALSARawMidiInputPort::~JackALSARawMidiInputPort()
  40. {
  41. delete raw_queue;
  42. delete receive_queue;
  43. delete thread_queue;
  44. delete write_queue;
  45. }
  46. jack_nframes_t
  47. JackALSARawMidiInputPort::EnqueueALSAEvent()
  48. {
  49. switch (raw_queue->EnqueueEvent(alsa_event)) {
  50. case JackMidiWriteQueue::BUFFER_FULL:
  51. // Processing events early might free up some space in the raw queue.
  52. raw_queue->Process();
  53. switch (raw_queue->EnqueueEvent(alsa_event)) {
  54. case JackMidiWriteQueue::BUFFER_TOO_SMALL:
  55. jack_error("JackALSARawMidiInputPort::Process - **BUG** "
  56. "JackMidiRawInputWriteQueue::EnqueueEvent returned "
  57. "`BUFFER_FULL` and then returned `BUFFER_TOO_SMALL` "
  58. "after a `Process()` call.");
  59. // Fallthrough on purpose
  60. case JackMidiWriteQueue::OK:
  61. return 0;
  62. default:
  63. ;
  64. }
  65. break;
  66. case JackMidiWriteQueue::BUFFER_TOO_SMALL:
  67. jack_error("JackALSARawMidiInputPort::Execute - The thread queue "
  68. "couldn't enqueue a %d-byte packet. Dropping event.",
  69. alsa_event->size);
  70. // Fallthrough on purpose
  71. case JackMidiWriteQueue::OK:
  72. return 0;
  73. default:
  74. ;
  75. }
  76. jack_nframes_t alsa_time = alsa_event->time;
  77. jack_nframes_t next_time = raw_queue->Process();
  78. return (next_time < alsa_time) ? next_time : alsa_time;
  79. }
  80. bool
  81. JackALSARawMidiInputPort::ProcessALSA(jack_nframes_t *frame)
  82. {
  83. unsigned short revents;
  84. if (! ProcessPollEvents(&revents)) {
  85. return false;
  86. }
  87. if (alsa_event) {
  88. *frame = EnqueueALSAEvent();
  89. if (*frame) {
  90. return true;
  91. }
  92. }
  93. if (revents & POLLIN) {
  94. for (alsa_event = receive_queue->DequeueEvent(); alsa_event;
  95. alsa_event = receive_queue->DequeueEvent()) {
  96. *frame = EnqueueALSAEvent();
  97. if (*frame) {
  98. return true;
  99. }
  100. }
  101. }
  102. *frame = raw_queue->Process();
  103. return true;
  104. }
  105. bool
  106. JackALSARawMidiInputPort::ProcessJack(JackMidiBuffer *port_buffer,
  107. jack_nframes_t frames)
  108. {
  109. write_queue->ResetMidiBuffer(port_buffer, frames);
  110. if (! jack_event) {
  111. jack_event = thread_queue->DequeueEvent();
  112. }
  113. for (; jack_event; jack_event = thread_queue->DequeueEvent()) {
  114. // We add `frames` so that MIDI events align with audio as closely as
  115. // possible.
  116. switch (write_queue->EnqueueEvent(jack_event, frames)) {
  117. case JackMidiWriteQueue::BUFFER_TOO_SMALL:
  118. jack_error("JackALSARawMidiInputPort::ProcessJack - The write "
  119. "queue couldn't enqueue a %d-byte event. Dropping "
  120. "event.", jack_event->size);
  121. // Fallthrough on purpose
  122. case JackMidiWriteQueue::OK:
  123. continue;
  124. default:
  125. ;
  126. }
  127. break;
  128. }
  129. return true;
  130. }