ALSA backend: add raw midi parser
This commit is contained in:
parent
5cd2010c79
commit
c79a56a08f
@ -1,5 +1,6 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (C) 2014 Robin Gareus <robin@gareus.org>
|
* Copyright (C) 2014 Robin Gareus <robin@gareus.org>
|
||||||
|
* Copyright (C) 2010 Devin Anderson
|
||||||
*
|
*
|
||||||
* This program is free software; you can redistribute it and/or modify
|
* This program is free software; you can redistribute it and/or modify
|
||||||
* it under the terms of the GNU General Public License as published by
|
* it under the terms of the GNU General Public License as published by
|
||||||
@ -335,6 +336,11 @@ retry:
|
|||||||
|
|
||||||
AlsaRawMidiIn::AlsaRawMidiIn (const char *device)
|
AlsaRawMidiIn::AlsaRawMidiIn (const char *device)
|
||||||
: AlsaRawMidiIO (device, true)
|
: AlsaRawMidiIO (device, true)
|
||||||
|
, _event(0,0)
|
||||||
|
, _unbuffered_bytes(0)
|
||||||
|
, _total_bytes(0)
|
||||||
|
, _expected_bytes(0)
|
||||||
|
, _status_byte(0)
|
||||||
{
|
{
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -450,23 +456,136 @@ AlsaRawMidiIn::main_process_thread ()
|
|||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!(data[0] & 0x80)) {
|
#if 0
|
||||||
_DEBUGPRINT("AlsaRawMidiIn: invalid midi message.\n");
|
queue_event (time, data, err);
|
||||||
}
|
#else
|
||||||
// TODO parse MIDI-events? break on status-bytes
|
parse_events (time, data, err);
|
||||||
{
|
#endif
|
||||||
ssize_t size = err;
|
|
||||||
const uint32_t buf_size = sizeof(MidiEventHeader) + size;
|
|
||||||
if (_rb->write_space() < buf_size) {
|
|
||||||
_DEBUGPRINT("AlsaRawMidiIn: ring buffer overflow\n");
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
struct MidiEventHeader h (time, size);
|
|
||||||
_rb->write ((uint8_t*) &h, sizeof(MidiEventHeader));
|
|
||||||
_rb->write (data, size);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_DEBUGPRINT("AlsaRawMidiIn: MIDI IN THREAD STOPPED\n");
|
_DEBUGPRINT("AlsaRawMidiIn: MIDI IN THREAD STOPPED\n");
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
int
|
||||||
|
AlsaRawMidiIn::queue_event (const uint64_t time, const uint8_t *data, const size_t size) {
|
||||||
|
const uint32_t buf_size = sizeof(MidiEventHeader) + size;
|
||||||
|
_event._pending = false;
|
||||||
|
if (_rb->write_space() < buf_size) {
|
||||||
|
_DEBUGPRINT("AlsaRawMidiIn: ring buffer overflow\n");
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
struct MidiEventHeader h (time, size);
|
||||||
|
_rb->write ((uint8_t*) &h, sizeof(MidiEventHeader));
|
||||||
|
_rb->write (data, size);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
void
|
||||||
|
AlsaRawMidiIn::parse_events (const uint64_t time, const uint8_t *data, const size_t size) {
|
||||||
|
if (_event._pending) {
|
||||||
|
if (queue_event (_event._time, _parser_buffer, _event._size)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
for (size_t i = 0; i < size; ++i) {
|
||||||
|
if (process_byte(time, data[i])) {
|
||||||
|
if (queue_event (_event._time, _parser_buffer, _event._size)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// based on JackMidiRawInputWriteQueue by Devin Anderson //
|
||||||
|
bool
|
||||||
|
AlsaRawMidiIn::process_byte(const uint64_t time, const uint8_t byte)
|
||||||
|
{
|
||||||
|
if (byte >= 0xf8) {
|
||||||
|
// Realtime
|
||||||
|
if (byte == 0xfd) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
_parser_buffer[0] = byte;
|
||||||
|
prepare_byte_event(time, byte);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (byte == 0xf7) {
|
||||||
|
// Sysex end
|
||||||
|
if (_status_byte == 0xf0) {
|
||||||
|
record_byte(byte);
|
||||||
|
return prepare_buffered_event(time);
|
||||||
|
}
|
||||||
|
_total_bytes = 0;
|
||||||
|
_unbuffered_bytes = 0;
|
||||||
|
_expected_bytes = 0;
|
||||||
|
_status_byte = 0;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (byte >= 0x80) {
|
||||||
|
// Non-realtime status byte
|
||||||
|
if (_total_bytes) {
|
||||||
|
_total_bytes = 0;
|
||||||
|
_unbuffered_bytes = 0;
|
||||||
|
}
|
||||||
|
_status_byte = byte;
|
||||||
|
switch (byte & 0xf0) {
|
||||||
|
case 0x80:
|
||||||
|
case 0x90:
|
||||||
|
case 0xa0:
|
||||||
|
case 0xb0:
|
||||||
|
case 0xe0:
|
||||||
|
// Note On, Note Off, Aftertouch, Control Change, Pitch Wheel
|
||||||
|
_expected_bytes = 3;
|
||||||
|
break;
|
||||||
|
case 0xc0:
|
||||||
|
case 0xd0:
|
||||||
|
// Program Change, Channel Pressure
|
||||||
|
_expected_bytes = 2;
|
||||||
|
break;
|
||||||
|
case 0xf0:
|
||||||
|
switch (byte) {
|
||||||
|
case 0xf0:
|
||||||
|
// Sysex
|
||||||
|
_expected_bytes = 0;
|
||||||
|
break;
|
||||||
|
case 0xf1:
|
||||||
|
case 0xf3:
|
||||||
|
// MTC Quarter Frame, Song Select
|
||||||
|
_expected_bytes = 2;
|
||||||
|
break;
|
||||||
|
case 0xf2:
|
||||||
|
// Song Position
|
||||||
|
_expected_bytes = 3;
|
||||||
|
break;
|
||||||
|
case 0xf4:
|
||||||
|
case 0xf5:
|
||||||
|
// Undefined
|
||||||
|
_expected_bytes = 0;
|
||||||
|
_status_byte = 0;
|
||||||
|
return false;
|
||||||
|
case 0xf6:
|
||||||
|
// Tune Request
|
||||||
|
prepare_byte_event(time, byte);
|
||||||
|
_expected_bytes = 0;
|
||||||
|
_status_byte = 0;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
record_byte(byte);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
// Data byte
|
||||||
|
if (! _status_byte) {
|
||||||
|
// Data bytes without a status will be discarded.
|
||||||
|
_total_bytes++;
|
||||||
|
_unbuffered_bytes++;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (! _total_bytes) {
|
||||||
|
// Apply running status.
|
||||||
|
record_byte(_status_byte);
|
||||||
|
}
|
||||||
|
record_byte(byte);
|
||||||
|
return (_total_bytes == _expected_bytes) ? prepare_buffered_event(time) : false;
|
||||||
|
}
|
||||||
|
@ -93,6 +93,61 @@ public:
|
|||||||
void* main_process_thread ();
|
void* main_process_thread ();
|
||||||
|
|
||||||
size_t recv_event (pframes_t &, uint8_t *, size_t &);
|
size_t recv_event (pframes_t &, uint8_t *, size_t &);
|
||||||
|
|
||||||
|
private:
|
||||||
|
int queue_event (const uint64_t, const uint8_t *, const size_t);
|
||||||
|
void parse_events (const uint64_t, const uint8_t *, const size_t);
|
||||||
|
bool process_byte (const uint64_t, const uint8_t);
|
||||||
|
|
||||||
|
void record_byte(uint8_t byte) {
|
||||||
|
if (_total_bytes < sizeof(_parser_buffer)) {
|
||||||
|
_parser_buffer[_total_bytes] = byte;
|
||||||
|
} else {
|
||||||
|
++_unbuffered_bytes;
|
||||||
|
}
|
||||||
|
++_total_bytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
void prepare_byte_event(const uint64_t time, const uint8_t byte) {
|
||||||
|
_parser_buffer[0] = byte;
|
||||||
|
_event.prepare(time, 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
bool prepare_buffered_event(const uint64_t time) {
|
||||||
|
const bool result = !_unbuffered_bytes;
|
||||||
|
if (result) {
|
||||||
|
_event.prepare(time, _total_bytes);
|
||||||
|
}
|
||||||
|
_total_bytes = 0;
|
||||||
|
_unbuffered_bytes = 0;
|
||||||
|
if (_status_byte >= 0xf0) {
|
||||||
|
_expected_bytes = 0;
|
||||||
|
_status_byte = 0;
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
struct ParserEvent {
|
||||||
|
uint64_t _time;
|
||||||
|
size_t _size;
|
||||||
|
bool _pending;
|
||||||
|
ParserEvent (const uint64_t time, const size_t size)
|
||||||
|
: _time(time)
|
||||||
|
, _size(size)
|
||||||
|
, _pending(false) {}
|
||||||
|
|
||||||
|
void prepare(const uint64_t time, const size_t size) {
|
||||||
|
_time = time;
|
||||||
|
_size = size;
|
||||||
|
_pending = true;
|
||||||
|
}
|
||||||
|
} _event;
|
||||||
|
|
||||||
|
size_t _unbuffered_bytes;
|
||||||
|
size_t _total_bytes;
|
||||||
|
size_t _expected_bytes;
|
||||||
|
uint8_t _status_byte;
|
||||||
|
uint8_t _parser_buffer[1024];
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace
|
} // namespace
|
||||||
|
Loading…
Reference in New Issue
Block a user