src/JackCommand.h
author František Kučera <franta-hg@frantovo.cz>
Tue, 29 Sep 2020 22:53:08 +0200
branchv_0
changeset 8 8ef1980db907
parent 6 b81bff3ebc4c
child 9 0d362165241e
permissions -rw-r--r--
configurable JACK client name
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
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:
8
8ef1980db907 configurable JACK client name
František Kučera <franta-hg@frantovo.cz>
parents: 6
diff changeset
    53
	Configuration& configuration;
8ef1980db907 configurable JACK client name
František Kučera <franta-hg@frantovo.cz>
parents: 6
diff changeset
    54
	std::wstring_convert<std::codecvt_utf8<wchar_t>> convertor; // TODO: local system encoding
8ef1980db907 configurable JACK client name
František Kučera <franta-hg@frantovo.cz>
parents: 6
diff changeset
    55
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    56
	jack_port_t* jackPort = nullptr;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    57
	jack_ringbuffer_t* ringBuffer = nullptr;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    58
	pthread_mutex_t messageThreadLock = PTHREAD_MUTEX_INITIALIZER;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    59
	pthread_cond_t dataReady = PTHREAD_COND_INITIALIZER;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    60
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    61
	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
    62
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    63
	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
    64
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    65
	struct MidiMessage {
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    66
		uint8_t buffer[4096];
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    67
		uint32_t size;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    68
		uint32_t time;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    69
	};
3
0222c20f590f report also unknown messages
František Kučera <franta-hg@frantovo.cz>
parents: 2
diff changeset
    70
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    71
public:
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    72
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    73
	int enqueueMessage(jack_nframes_t frames) {
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    74
		void* buffer = jack_port_get_buffer(jackPort, frames);
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    75
		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
    76
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    77
		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
    78
			jack_midi_event_t event;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    79
			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
    80
			if (noData) continue;
0
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
    81
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    82
			if (event.size > sizeof (MidiMessage::buffer)) {
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    83
				// TODO: should not printf in RT callback:
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    84
				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
    85
			} 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
    86
				MidiMessage m;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    87
				m.time = event.time;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    88
				m.size = event.size;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    89
				memcpy(m.buffer, event.buffer, event.size);
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    90
				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
    91
			} else {
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    92
				// TODO: should not printf in RT callback:
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    93
				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
    94
			}
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    95
		}
0
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
    96
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    97
		// 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
    98
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
    99
		if (pthread_mutex_trylock(&messageThreadLock) == 0) {
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   100
			pthread_cond_signal(&dataReady);
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   101
			pthread_mutex_unlock(&messageThreadLock);
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
		return 0;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   105
	}
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   106
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   107
private:
0
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   108
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   109
	static void writeRecord(std::shared_ptr<RelationalWriter> writer,
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   110
			string_t eventType, integer_t channel,
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   111
			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
   112
			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
   113
			string_t raw) {
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   114
		writer->writeAttribute(eventType);
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   115
		writer->writeAttribute(&channel, typeid (channel));
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   116
		writer->writeAttribute(&noteOn, typeid (noteOn));
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   117
		writer->writeAttribute(&pitch, typeid (pitch));
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   118
		writer->writeAttribute(&velocity, typeid (velocity));
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   119
		writer->writeAttribute(&controllerId, typeid (controllerId));
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   120
		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
   121
		writer->writeAttribute(&raw, typeid (raw));
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   122
	}
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   123
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   124
	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
   125
		if (event->size == 0) {
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   126
			return;
0
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   127
		} else {
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   128
			uint8_t type = event->buffer[0] & 0xF0;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   129
			uint8_t channel = event->buffer[0] & 0x0F;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   130
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   131
			// TODO: write timestamp, message number
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   132
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   133
			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
   134
				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
   135
			} 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
   136
				writeRecord(writer, L"control", channel, false, 0, 0, event->buffer[1], event->buffer[2], toHex(event));
5
40dd6deafaca recognize SysEx (System Exclusive) messages
František Kučera <franta-hg@frantovo.cz>
parents: 4
diff changeset
   137
			} else if (event->buffer[0] == 0xF0) {
40dd6deafaca recognize SysEx (System Exclusive) messages
František Kučera <franta-hg@frantovo.cz>
parents: 4
diff changeset
   138
				writeRecord(writer, L"sysex", channel, false, 0, 0, 0, 0, toHex(event));
3
0222c20f590f report also unknown messages
František Kučera <franta-hg@frantovo.cz>
parents: 2
diff changeset
   139
			} else {
6
b81bff3ebc4c fix unknown message output
František Kučera <franta-hg@frantovo.cz>
parents: 5
diff changeset
   140
				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
   141
			}
0
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   142
		}
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   143
	}
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   144
4
30da4232cdbc add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents: 3
diff changeset
   145
	string_t toHex(MidiMessage* event) {
30da4232cdbc add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents: 3
diff changeset
   146
		std::wstringstream result;
30da4232cdbc add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents: 3
diff changeset
   147
		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
   148
30da4232cdbc add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents: 3
diff changeset
   149
		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
   150
			if (i > 0) result << L' ';
30da4232cdbc add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents: 3
diff changeset
   151
			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
   152
			// 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
   153
			// 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
   154
		}
30da4232cdbc add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents: 3
diff changeset
   155
30da4232cdbc add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents: 3
diff changeset
   156
		return result.str();
30da4232cdbc add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents: 3
diff changeset
   157
	}
30da4232cdbc add raw MIDI data in HEX
František Kučera <franta-hg@frantovo.cz>
parents: 3
diff changeset
   158
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   159
public:
0
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   160
8
8ef1980db907 configurable JACK client name
František Kučera <franta-hg@frantovo.cz>
parents: 6
diff changeset
   161
	JackCommand(Configuration& configuration) : configuration(configuration) {
8ef1980db907 configurable JACK client name
František Kučera <franta-hg@frantovo.cz>
parents: 6
diff changeset
   162
	}
8ef1980db907 configurable JACK client name
František Kučera <franta-hg@frantovo.cz>
parents: 6
diff changeset
   163
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   164
	void finish(int sig) {
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   165
		continueProcessing = false;
0
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   166
	}
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   167
2
e5f0d3f92eb4 pass RelationalWriter instead of std::ostream to the JackCommand
František Kučera <franta-hg@frantovo.cz>
parents: 1
diff changeset
   168
	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
   169
		// Relation headers:
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   170
		vector<AttributeMetadata> metadata;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   171
		metadata.push_back({L"event", TypeId::STRING});
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   172
		metadata.push_back({L"channel", TypeId::INTEGER});
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   173
		metadata.push_back({L"note_on", TypeId::BOOLEAN});
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   174
		metadata.push_back({L"note_pitch", TypeId::INTEGER});
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   175
		metadata.push_back({L"note_velocity", TypeId::INTEGER});
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   176
		metadata.push_back({L"controller_id", TypeId::INTEGER});
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   177
		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
   178
		metadata.push_back({L"raw", TypeId::STRING});
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   179
		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
   180
		relationalWriterFlush();
0
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   181
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   182
		// Initialize JACK connection:
8
8ef1980db907 configurable JACK client name
František Kučera <franta-hg@frantovo.cz>
parents: 6
diff changeset
   183
		std::string clientName = convertor.to_bytes(configuration.jackClientName);
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   184
		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
   185
		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
   186
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   187
		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
   188
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   189
		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
   190
		// TODO: report also other events (connections etc.)
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   191
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   192
		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
   193
		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
   194
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   195
		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
   196
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   197
		int jackError = jack_activate(client);
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   198
		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
   199
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   200
		// Process messages from the ring buffer queue:
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   201
		pthread_mutex_lock(&messageThreadLock);
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   202
		while (continueProcessing) {
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   203
			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
   204
			for (size_t i = 0; i < queuedMessages; ++i) {
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   205
				MidiMessage m;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   206
				jack_ringbuffer_read(ringBuffer, (char*) &m, sizeof (MidiMessage));
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   207
				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
   208
				relationalWriterFlush();
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   209
			}
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   210
			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
   211
		}
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   212
		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
   213
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   214
		// Close JACK connection:
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   215
		jack_deactivate(client);
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   216
		jack_client_close(client);
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   217
		jack_ringbuffer_free(ringBuffer);
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   218
	}
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   219
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   220
};
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   221
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   222
int enqueueMessage(jack_nframes_t frames, void* arg) {
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   223
	JackCommand* instance = (JackCommand*) arg;
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   224
	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
   225
}
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   226
1
001b956610ca separate JackCommand class
František Kučera <franta-hg@frantovo.cz>
parents: 0
diff changeset
   227
}
0
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   228
}
c8c8ec34120f first dirty version, writes MIDI events from JACK
František Kučera <franta-hg@frantovo.cz>
parents:
diff changeset
   229
}