author | František Kučera <franta-hg@frantovo.cz> |
Mon, 18 May 2020 23:37:38 +0200 | |
branch | v_0 |
changeset 4 | 30da4232cdbc |
parent 3 | 0222c20f590f |
child 5 | 40dd6deafaca |
permissions | -rw-r--r-- |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
1 |
/** |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
2 |
* Relational pipes |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
3 |
* Copyright © 2020 František Kučera (Frantovo.cz, GlobalCode.info) |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
4 |
* |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
5 |
* This program is free software: you can redistribute it and/or modify |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
6 |
* it under the terms of the GNU General Public License as published by |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
7 |
* the Free Software Foundation, version 3 of the License. |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
8 |
* |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
9 |
* This program is distributed in the hope that it will be useful, |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
10 |
* but WITHOUT ANY WARRANTY; without even the implied warranty of |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
11 |
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
12 |
* GNU General Public License for more details. |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
13 |
* |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
14 |
* You should have received a copy of the GNU General Public License |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
15 |
* along with this program. If not, see <http://www.gnu.org/licenses/>. |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
16 |
*/ |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
17 |
#pragma once |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
18 |
|
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
19 |
#include <cstdlib> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
20 |
#include <cstring> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
21 |
#include <memory> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
22 |
#include <unistd.h> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
23 |
#include <pthread.h> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
24 |
#include <sys/mman.h> |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
25 |
#include <atomic> |
2
e5f0d3f92eb4
pass RelationalWriter instead of std::ostream to the JackCommand
František Kučera <franta-hg@frantovo.cz>
parents:
1
diff
changeset
|
26 |
#include <functional> |
4
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
27 |
#include <sstream> |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
28 |
#include <iomanip> |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
29 |
|
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
30 |
#include <jack/jack.h> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
31 |
#include <jack/midiport.h> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
32 |
#include <jack/ringbuffer.h> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
33 |
|
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
34 |
#include <relpipe/writer/RelationalWriter.h> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
35 |
#include <relpipe/writer/RelpipeWriterException.h> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
36 |
#include <relpipe/writer/AttributeMetadata.h> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
37 |
#include <relpipe/writer/Factory.h> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
38 |
#include <relpipe/writer/TypeId.h> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
39 |
#include <relpipe/cli/CLI.h> |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
40 |
|
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
41 |
#include "JackException.h" |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
42 |
|
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
43 |
using namespace relpipe::writer; |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
44 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
45 |
namespace relpipe { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
46 |
namespace in { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
47 |
namespace jack { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
48 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
49 |
int enqueueMessage(jack_nframes_t frames, void* arg); |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
50 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
51 |
class JackCommand { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
52 |
private: |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
53 |
jack_port_t* jackPort = nullptr; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
54 |
jack_ringbuffer_t* ringBuffer = nullptr; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
55 |
pthread_mutex_t messageThreadLock = PTHREAD_MUTEX_INITIALIZER; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
56 |
pthread_cond_t dataReady = PTHREAD_COND_INITIALIZER; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
57 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
58 |
std::atomic<bool> continueProcessing{true}; |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
59 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
60 |
const int RING_BUFFER_SIZE = 100; |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
61 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
62 |
struct MidiMessage { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
63 |
uint8_t buffer[4096]; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
64 |
uint32_t size; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
65 |
uint32_t time; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
66 |
}; |
3
0222c20f590f
report also unknown messages
František Kučera <franta-hg@frantovo.cz>
parents:
2
diff
changeset
|
67 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
68 |
public: |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
69 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
70 |
int enqueueMessage(jack_nframes_t frames) { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
71 |
void* buffer = jack_port_get_buffer(jackPort, frames); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
72 |
if (buffer == nullptr) throw JackException(L"Unable to get port buffer."); // TODO: exception in RT callback? |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
73 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
74 |
for (jack_nframes_t i = 0, eventCount = jack_midi_get_event_count(buffer); i < eventCount; i++) { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
75 |
jack_midi_event_t event; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
76 |
int noData = jack_midi_event_get(&event, buffer, i); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
77 |
if (noData) continue; |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
78 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
79 |
if (event.size > sizeof (MidiMessage::buffer)) { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
80 |
// TODO: should not printf in RT callback: |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
81 |
fwprintf(stderr, L"Error: MIDI message was too large → skipping event. Maximum allowed size: %lu bytes.\n", sizeof (MidiMessage::buffer)); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
82 |
} else if (jack_ringbuffer_write_space(ringBuffer) >= sizeof (MidiMessage)) { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
83 |
MidiMessage m; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
84 |
m.time = event.time; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
85 |
m.size = event.size; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
86 |
memcpy(m.buffer, event.buffer, event.size); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
87 |
jack_ringbuffer_write(ringBuffer, (const char *) &m, sizeof (MidiMessage)); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
88 |
} else { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
89 |
// TODO: should not printf in RT callback: |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
90 |
fwprintf(stderr, L"Error: ring buffer is full → skipping event.\n"); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
91 |
} |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
92 |
} |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
93 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
94 |
// TODO: just count skipped events and bytes and report them in next successful message instead of printing to STDERR |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
95 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
96 |
if (pthread_mutex_trylock(&messageThreadLock) == 0) { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
97 |
pthread_cond_signal(&dataReady); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
98 |
pthread_mutex_unlock(&messageThreadLock); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
99 |
} |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
100 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
101 |
return 0; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
102 |
} |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
103 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
104 |
private: |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
105 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
106 |
static void writeRecord(std::shared_ptr<RelationalWriter> writer, |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
107 |
string_t eventType, integer_t channel, |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
108 |
boolean_t noteOn, integer_t pitch, integer_t velocity, |
4
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
109 |
integer_t controllerId, integer_t value, |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
110 |
string_t raw) { |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
111 |
writer->writeAttribute(eventType); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
112 |
writer->writeAttribute(&channel, typeid (channel)); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
113 |
writer->writeAttribute(¬eOn, typeid (noteOn)); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
114 |
writer->writeAttribute(&pitch, typeid (pitch)); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
115 |
writer->writeAttribute(&velocity, typeid (velocity)); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
116 |
writer->writeAttribute(&controllerId, typeid (controllerId)); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
117 |
writer->writeAttribute(&value, typeid (value)); |
4
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
118 |
writer->writeAttribute(&raw, typeid (raw)); |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
119 |
} |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
120 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
121 |
void processMessage(std::shared_ptr<RelationalWriter> writer, MidiMessage* event) { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
122 |
if (event->size == 0) { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
123 |
return; |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
124 |
} else { |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
125 |
uint8_t type = event->buffer[0] & 0xF0; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
126 |
uint8_t channel = event->buffer[0] & 0x0F; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
127 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
128 |
// TODO: write timestamp, message number |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
129 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
130 |
if ((type == 0x90 || type == 0x80) && event->size == 3) { |
4
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
131 |
writeRecord(writer, L"note", channel, type == 0x90, event->buffer[1], event->buffer[2], 0, 0, toHex(event)); |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
132 |
} else if (type == 0xB0 && event->size == 3) { |
4
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
133 |
writeRecord(writer, L"control", channel, false, 0, 0, event->buffer[1], event->buffer[2], toHex(event)); |
3
0222c20f590f
report also unknown messages
František Kučera <franta-hg@frantovo.cz>
parents:
2
diff
changeset
|
134 |
} else { |
4
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
135 |
writeRecord(writer, L"unknown", channel, false, 0, 0, 0, 0, toHex(event)); |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
136 |
} |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
137 |
} |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
138 |
} |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
139 |
|
4
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
140 |
string_t toHex(MidiMessage* event) { |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
141 |
std::wstringstream result; |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
142 |
result << std::hex << std::setfill(L'0'); |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
143 |
|
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
144 |
for (size_t i = 0; i < event->size && i < sizeof (event->buffer); i++) { |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
145 |
if (i > 0) result << L' '; |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
146 |
result << std::setw(2) << event->buffer[i]; |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
147 |
// result << ("0123456789abcdef"[event->buffer[i] >> 4]); |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
148 |
// result << ("0123456789abcdef"[event->buffer[i] & 0xf]); |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
149 |
} |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
150 |
|
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
151 |
return result.str(); |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
152 |
} |
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
153 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
154 |
public: |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
155 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
156 |
void finish(int sig) { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
157 |
continueProcessing = false; |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
158 |
} |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
159 |
|
2
e5f0d3f92eb4
pass RelationalWriter instead of std::ostream to the JackCommand
František Kučera <franta-hg@frantovo.cz>
parents:
1
diff
changeset
|
160 |
void processJackStream(std::shared_ptr<writer::RelationalWriter> writer, std::function<void() > relationalWriterFlush) { |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
161 |
// Relation headers: |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
162 |
vector<AttributeMetadata> metadata; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
163 |
metadata.push_back({L"event", TypeId::STRING}); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
164 |
metadata.push_back({L"channel", TypeId::INTEGER}); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
165 |
metadata.push_back({L"note_on", TypeId::BOOLEAN}); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
166 |
metadata.push_back({L"note_pitch", TypeId::INTEGER}); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
167 |
metadata.push_back({L"note_velocity", TypeId::INTEGER}); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
168 |
metadata.push_back({L"controller_id", TypeId::INTEGER}); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
169 |
metadata.push_back({L"controller_value", TypeId::INTEGER}); |
4
30da4232cdbc
add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents:
3
diff
changeset
|
170 |
metadata.push_back({L"raw", TypeId::STRING}); |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
171 |
writer->startRelation(L"midi", metadata, true); |
2
e5f0d3f92eb4
pass RelationalWriter instead of std::ostream to the JackCommand
František Kučera <franta-hg@frantovo.cz>
parents:
1
diff
changeset
|
172 |
relationalWriterFlush(); |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
173 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
174 |
// Initialize JACK connection: |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
175 |
std::string clientName = "relpipe-in-jack"; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
176 |
jack_client_t* client = jack_client_open(clientName.c_str(), JackNullOption, nullptr); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
177 |
if (client == nullptr) throw JackException(L"Could not create JACK client."); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
178 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
179 |
ringBuffer = jack_ringbuffer_create(RING_BUFFER_SIZE * sizeof (MidiMessage)); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
180 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
181 |
jack_set_process_callback(client, relpipe::in::jack::enqueueMessage, this); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
182 |
// TODO: report also other events (connections etc.) |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
183 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
184 |
jackPort = jack_port_register(client, "input", JACK_DEFAULT_MIDI_TYPE, JackPortIsInput, 0); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
185 |
if (jackPort == nullptr) throw JackException(L"Could not register port."); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
186 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
187 |
if (mlockall(MCL_CURRENT | MCL_FUTURE)) fwprintf(stderr, L"Warning: Can not lock memory.\n"); |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
188 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
189 |
int jackError = jack_activate(client); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
190 |
if (jackError) throw JackException(L"Could not activate client."); |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
191 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
192 |
// Process messages from the ring buffer queue: |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
193 |
pthread_mutex_lock(&messageThreadLock); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
194 |
while (continueProcessing) { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
195 |
const size_t queuedMessages = jack_ringbuffer_read_space(ringBuffer) / sizeof (MidiMessage); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
196 |
for (size_t i = 0; i < queuedMessages; ++i) { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
197 |
MidiMessage m; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
198 |
jack_ringbuffer_read(ringBuffer, (char*) &m, sizeof (MidiMessage)); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
199 |
processMessage(writer, &m); |
2
e5f0d3f92eb4
pass RelationalWriter instead of std::ostream to the JackCommand
František Kučera <franta-hg@frantovo.cz>
parents:
1
diff
changeset
|
200 |
relationalWriterFlush(); |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
201 |
} |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
202 |
pthread_cond_wait(&dataReady, &messageThreadLock); |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
203 |
} |
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
204 |
pthread_mutex_unlock(&messageThreadLock); |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
205 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
206 |
// Close JACK connection: |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
207 |
jack_deactivate(client); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
208 |
jack_client_close(client); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
209 |
jack_ringbuffer_free(ringBuffer); |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
210 |
} |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
211 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
212 |
}; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
213 |
|
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
214 |
int enqueueMessage(jack_nframes_t frames, void* arg) { |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
215 |
JackCommand* instance = (JackCommand*) arg; |
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
216 |
return instance->enqueueMessage(frames); |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
217 |
} |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
218 |
|
1
001b956610ca
separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents:
0
diff
changeset
|
219 |
} |
0
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
220 |
} |
c8c8ec34120f
first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff
changeset
|
221 |
} |