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.

125 lines
4.1KB

  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 <cassert>
  16. #include <memory>
  17. #include "JackALSARawMidiInputPort.h"
  18. #include "JackMidiUtil.h"
  19. #include "JackError.h"
  20. using Jack::JackALSARawMidiInputPort;
  21. JackALSARawMidiInputPort::JackALSARawMidiInputPort(const char* client_name,
  22. snd_rawmidi_info_t *info,
  23. size_t index,
  24. size_t max_bytes,
  25. size_t max_messages):
  26. JackALSARawMidiPort(client_name, info, index, POLLIN)
  27. {
  28. alsa_event = 0;
  29. jack_event = 0;
  30. receive_queue = new JackALSARawMidiReceiveQueue(rawmidi, max_bytes);
  31. std::unique_ptr<JackALSARawMidiReceiveQueue> receive_ptr(receive_queue);
  32. thread_queue = new JackMidiAsyncQueue(max_bytes, max_messages);
  33. std::unique_ptr<JackMidiAsyncQueue> thread_ptr(thread_queue);
  34. write_queue = new JackMidiBufferWriteQueue();
  35. std::unique_ptr<JackMidiBufferWriteQueue> write_ptr(write_queue);
  36. raw_queue = new JackMidiRawInputWriteQueue(thread_queue, max_bytes,
  37. max_messages);
  38. write_ptr.release();
  39. thread_ptr.release();
  40. receive_ptr.release();
  41. }
  42. JackALSARawMidiInputPort::~JackALSARawMidiInputPort()
  43. {
  44. delete raw_queue;
  45. delete receive_queue;
  46. delete thread_queue;
  47. delete write_queue;
  48. }
  49. bool
  50. JackALSARawMidiInputPort::ProcessJack(JackMidiBuffer *port_buffer,
  51. jack_nframes_t frames)
  52. {
  53. write_queue->ResetMidiBuffer(port_buffer, frames);
  54. bool dequeued = false;
  55. if (! jack_event) {
  56. goto dequeue_event;
  57. }
  58. for (;;) {
  59. switch (write_queue->EnqueueEvent(jack_event, frames)) {
  60. case JackMidiWriteQueue::BUFFER_TOO_SMALL:
  61. jack_error("JackALSARawMidiInputPort::ProcessJack - The write "
  62. "queue couldn't enqueue a %d-byte event. Dropping "
  63. "event.", jack_event->size);
  64. // Fallthrough on purpose.
  65. case JackMidiWriteQueue::OK:
  66. break;
  67. default:
  68. goto trigger_queue_event;
  69. }
  70. dequeue_event:
  71. jack_event = thread_queue->DequeueEvent();
  72. if (! jack_event) {
  73. break;
  74. }
  75. dequeued = true;
  76. }
  77. trigger_queue_event:
  78. return dequeued ? TriggerQueueEvent() : true;
  79. }
  80. bool
  81. JackALSARawMidiInputPort::ProcessPollEvents(jack_nframes_t current_frame)
  82. {
  83. if (GetQueuePollEvent() == -1) {
  84. return false;
  85. }
  86. int io_event = GetIOPollEvent();
  87. switch (io_event) {
  88. case -1:
  89. return false;
  90. case 1:
  91. alsa_event = receive_queue->DequeueEvent();
  92. }
  93. if (alsa_event) {
  94. size_t size = alsa_event->size;
  95. size_t space = raw_queue->GetAvailableSpace();
  96. bool enough_room = space >= size;
  97. if (enough_room) {
  98. assert(raw_queue->EnqueueEvent(current_frame, size,
  99. alsa_event->buffer) ==
  100. JackMidiWriteQueue::OK);
  101. alsa_event = 0;
  102. } else if (space) {
  103. assert(raw_queue->EnqueueEvent(current_frame, space,
  104. alsa_event->buffer) ==
  105. JackMidiWriteQueue::OK);
  106. alsa_event->buffer += space;
  107. alsa_event->size -= space;
  108. }
  109. SetIOEventsEnabled(enough_room);
  110. }
  111. raw_queue->Process();
  112. return true;
  113. }