mirror of
https://github.com/Ardour/ardour.git
synced 2025-12-11 17:16:38 +01:00
Coreaudio backend RT-safe MIDI buffer allocation
This commit is contained in:
parent
59a63a08f9
commit
f4c76f89d3
2 changed files with 30 additions and 34 deletions
|
|
@ -114,6 +114,8 @@ CoreAudioBackend::CoreAudioBackend (AudioEngine& e, AudioBackendInfo& info)
|
||||||
pthread_mutex_init (&_freewheel_mutex, 0);
|
pthread_mutex_init (&_freewheel_mutex, 0);
|
||||||
pthread_cond_init (&_freewheel_signal, 0);
|
pthread_cond_init (&_freewheel_signal, 0);
|
||||||
|
|
||||||
|
_port_connection_queue.reserve (128);
|
||||||
|
|
||||||
_pcmio = new CoreAudioPCM ();
|
_pcmio = new CoreAudioPCM ();
|
||||||
_midiio = new CoreMidiIo ();
|
_midiio = new CoreMidiIo ();
|
||||||
|
|
||||||
|
|
@ -1415,11 +1417,11 @@ CoreAudioBackend::midi_event_get (
|
||||||
if (event_index >= source.size ()) {
|
if (event_index >= source.size ()) {
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
CoreMidiEvent * const event = source[event_index].get ();
|
CoreMidiEvent const& event = source[event_index].get ();
|
||||||
|
|
||||||
timestamp = event->timestamp ();
|
timestamp = event.timestamp ();
|
||||||
size = event->size ();
|
size = event.size ();
|
||||||
*buf = event->data ();
|
*buf = event.data ();
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -1430,15 +1432,18 @@ CoreAudioBackend::_midi_event_put (
|
||||||
const uint8_t* buffer, size_t size)
|
const uint8_t* buffer, size_t size)
|
||||||
{
|
{
|
||||||
if (!buffer || !port_buffer) return -1;
|
if (!buffer || !port_buffer) return -1;
|
||||||
|
if (size >= MaxCoreMidiEventSize) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
CoreMidiBuffer& dst = * static_cast<CoreMidiBuffer*>(port_buffer);
|
CoreMidiBuffer& dst = * static_cast<CoreMidiBuffer*>(port_buffer);
|
||||||
if (dst.size () && (pframes_t)dst.back ()->timestamp () > timestamp) {
|
|
||||||
#ifndef NDEBUG
|
#ifndef NDEBUG
|
||||||
|
if (dst.size () && (pframes_t)dst.back ().timestamp () > timestamp) {
|
||||||
// nevermind, ::get_buffer() sorts events
|
// nevermind, ::get_buffer() sorts events
|
||||||
fprintf (stderr, "CoreMidiBuffer: unordered event: %d > %d\n",
|
fprintf (stderr, "CoreMidiBuffer: unordered event: %d > %d\n",
|
||||||
(pframes_t)dst.back ()->timestamp (), timestamp);
|
(pframes_t)dst.back ().timestamp (), timestamp);
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
dst.push_back (boost::shared_ptr<CoreMidiEvent>(new CoreMidiEvent (timestamp, buffer, size)));
|
#endif
|
||||||
|
dst.push_back (CoreMidiEvent (timestamp, buffer, size));
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
@ -1802,7 +1807,7 @@ CoreAudioBackend::process_callback (const uint32_t n_samples, const uint64_t hos
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
uint64_t time_ns;
|
uint64_t time_ns;
|
||||||
uint8_t data[128]; // matches CoreMidi's MIDIPacket
|
uint8_t data[MaxCoreMidiEventSize];
|
||||||
size_t size = sizeof(data);
|
size_t size = sizeof(data);
|
||||||
|
|
||||||
port->clear_events ();
|
port->clear_events ();
|
||||||
|
|
@ -1845,15 +1850,10 @@ CoreAudioBackend::process_callback (const uint32_t n_samples, const uint64_t hos
|
||||||
/* queue outgoing midi */
|
/* queue outgoing midi */
|
||||||
i = 0;
|
i = 0;
|
||||||
for (std::vector<CoreBackendPort*>::const_iterator it = _system_midi_out.begin (); it != _system_midi_out.end (); ++it, ++i) {
|
for (std::vector<CoreBackendPort*>::const_iterator it = _system_midi_out.begin (); it != _system_midi_out.end (); ++it, ++i) {
|
||||||
#if 0 // something's still b0rked with CoreMidiIo::send_events()
|
|
||||||
const CoreMidiBuffer *src = static_cast<const CoreMidiPort*>(*it)->const_buffer();
|
|
||||||
_midiio->send_events (i, nominal_time, (void*)src);
|
|
||||||
#else // works..
|
|
||||||
const CoreMidiBuffer *src = static_cast<const CoreMidiPort*>(*it)->const_buffer();
|
const CoreMidiBuffer *src = static_cast<const CoreMidiPort*>(*it)->const_buffer();
|
||||||
for (CoreMidiBuffer::const_iterator mit = src->begin (); mit != src->end (); ++mit) {
|
for (CoreMidiBuffer::const_iterator mit = src->begin (); mit != src->end (); ++mit) {
|
||||||
_midiio->send_event (i, (*mit)->timestamp() / nominal_time, (*mit)->data(), (*mit)->size());
|
_midiio->send_event (i,tamp (), mit->data (), mit->size ());
|
||||||
}
|
}
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* write back audio */
|
/* write back audio */
|
||||||
|
|
@ -2159,13 +2159,16 @@ CoreMidiPort::CoreMidiPort (CoreAudioBackend &b, const std::string& name, PortFl
|
||||||
{
|
{
|
||||||
_buffer[0].clear ();
|
_buffer[0].clear ();
|
||||||
_buffer[1].clear ();
|
_buffer[1].clear ();
|
||||||
|
|
||||||
|
_buffer[0].reserve (256);
|
||||||
|
_buffer[1].reserve (256);
|
||||||
}
|
}
|
||||||
|
|
||||||
CoreMidiPort::~CoreMidiPort () { }
|
CoreMidiPort::~CoreMidiPort () { }
|
||||||
|
|
||||||
struct MidiEventSorter {
|
struct MidiEventSorter {
|
||||||
bool operator() (const boost::shared_ptr<CoreMidiEvent>& a, const boost::shared_ptr<CoreMidiEvent>& b) {
|
bool operator() (CoreMidiEvent const& a, CoreMidiEvent const& b) {
|
||||||
return *a < *b;
|
return a < b;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
@ -2179,7 +2182,7 @@ void* CoreMidiPort::get_buffer (pframes_t /* nframes */)
|
||||||
++i) {
|
++i) {
|
||||||
const CoreMidiBuffer * src = static_cast<const CoreMidiPort*>(*i)->const_buffer ();
|
const CoreMidiBuffer * src = static_cast<const CoreMidiPort*>(*i)->const_buffer ();
|
||||||
for (CoreMidiBuffer::const_iterator it = src->begin (); it != src->end (); ++it) {
|
for (CoreMidiBuffer::const_iterator it = src->begin (); it != src->end (); ++it) {
|
||||||
(_buffer[_bufperiod]).push_back (boost::shared_ptr<CoreMidiEvent>(new CoreMidiEvent (**it)));
|
(_buffer[_bufperiod]).push_back (*it);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
std::stable_sort ((_buffer[_bufperiod]).begin (), (_buffer[_bufperiod]).end (), MidiEventSorter());
|
std::stable_sort ((_buffer[_bufperiod]).begin (), (_buffer[_bufperiod]).end (), MidiEventSorter());
|
||||||
|
|
@ -2348,10 +2351,8 @@ CoreMidiPort::process_byte(const uint64_t time, const uint8_t byte)
|
||||||
CoreMidiEvent::CoreMidiEvent (const pframes_t timestamp, const uint8_t* data, size_t size)
|
CoreMidiEvent::CoreMidiEvent (const pframes_t timestamp, const uint8_t* data, size_t size)
|
||||||
: _size (size)
|
: _size (size)
|
||||||
, _timestamp (timestamp)
|
, _timestamp (timestamp)
|
||||||
, _data (0)
|
|
||||||
{
|
{
|
||||||
if (size > 0) {
|
if (size > 0 && size < MaxCoreMidiEventSize) {
|
||||||
_data = (uint8_t*) malloc (size);
|
|
||||||
memcpy (_data, data, size);
|
memcpy (_data, data, size);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
@ -2359,14 +2360,9 @@ CoreMidiEvent::CoreMidiEvent (const pframes_t timestamp, const uint8_t* data, si
|
||||||
CoreMidiEvent::CoreMidiEvent (const CoreMidiEvent& other)
|
CoreMidiEvent::CoreMidiEvent (const CoreMidiEvent& other)
|
||||||
: _size (other.size ())
|
: _size (other.size ())
|
||||||
, _timestamp (other.timestamp ())
|
, _timestamp (other.timestamp ())
|
||||||
, _data (0)
|
|
||||||
{
|
{
|
||||||
if (other.size () && other.const_data ()) {
|
if (other._size > 0) {
|
||||||
_data = (uint8_t*) malloc (other.size ());
|
assert (other._size < MaxCoreMidiEventSize);
|
||||||
memcpy (_data, other.const_data (), other.size ());
|
memcpy (_data, other._data, other._size);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
CoreMidiEvent::~CoreMidiEvent () {
|
|
||||||
free (_data);
|
|
||||||
};
|
|
||||||
|
|
|
||||||
|
|
@ -38,6 +38,8 @@
|
||||||
#include "coreaudio_pcmio.h"
|
#include "coreaudio_pcmio.h"
|
||||||
#include "coremidi_io.h"
|
#include "coremidi_io.h"
|
||||||
|
|
||||||
|
#define MaxCoreMidiEventSize 128 // matches CoreMidi's MIDIPacket
|
||||||
|
|
||||||
namespace ARDOUR {
|
namespace ARDOUR {
|
||||||
|
|
||||||
class CoreAudioBackend;
|
class CoreAudioBackend;
|
||||||
|
|
@ -46,19 +48,17 @@ class CoreMidiEvent {
|
||||||
public:
|
public:
|
||||||
CoreMidiEvent (const pframes_t timestamp, const uint8_t* data, size_t size);
|
CoreMidiEvent (const pframes_t timestamp, const uint8_t* data, size_t size);
|
||||||
CoreMidiEvent (const CoreMidiEvent& other);
|
CoreMidiEvent (const CoreMidiEvent& other);
|
||||||
~CoreMidiEvent ();
|
|
||||||
size_t size () const { return _size; };
|
size_t size () const { return _size; };
|
||||||
pframes_t timestamp () const { return _timestamp; };
|
pframes_t timestamp () const { return _timestamp; };
|
||||||
const unsigned char* const_data () const { return _data; };
|
const uint8_t* () const { return _data; };
|
||||||
unsigned char* data () { return _data; };
|
|
||||||
bool operator< (const CoreMidiEvent &other) const { return timestamp () < other.timestamp (); };
|
bool operator< (const CoreMidiEvent &other) const { return timestamp () < other.timestamp (); };
|
||||||
private:
|
private:
|
||||||
size_t _size;
|
size_t _size;
|
||||||
pframes_t _timestamp;
|
pframes_t _timestamp;
|
||||||
uint8_t *_data;
|
uint8_t _data[MaxCoreMidiEventSize];
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef std::vector<boost::shared_ptr<CoreMidiEvent> > CoreMidiBuffer;
|
typedef std::vector<CoreMidiEvent> CoreMidiBuffer;
|
||||||
|
|
||||||
class CoreBackendPort {
|
class CoreBackendPort {
|
||||||
protected:
|
protected:
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue