Optimize plugin-processing for non-automated params
[ardour.git] / libs / ardour / ardour / midi_buffer.h
index f1f8d678b979011321feb4a5437882b80dc6fed3..7793f7e0bd83946564f3e1925793398b1414faaa 100644 (file)
 #ifndef __ardour_midi_buffer_h__
 #define __ardour_midi_buffer_h__
 
+#include "evoral/EventSink.hpp"
 #include "evoral/midi_util.h"
+#include "evoral/types.hpp"
+
 #include "midi++/event.h"
+
 #include "ardour/buffer.h"
-#include "ardour/event_type_map.h"
+#include "ardour/parameter_types.h"
 
 namespace ARDOUR {
 
 
 /** Buffer containing 8-bit unsigned char (MIDI) data. */
-class LIBARDOUR_API MidiBuffer : public Buffer
+class LIBARDOUR_API MidiBuffer : public Buffer, public Evoral::EventSink<samplepos_t>
 {
 public:
-       typedef framepos_t TimeType;
+       typedef samplepos_t TimeType;
 
        MidiBuffer(size_t capacity);
        ~MidiBuffer();
 
-       void silence (framecnt_t nframes, framecnt_t offset = 0);
-       void read_from (const Buffer& src, framecnt_t nframes, framecnt_t dst_offset = 0, framecnt_t src_offset = 0);
-       void merge_from (const Buffer& src, framecnt_t nframes, framecnt_t dst_offset = 0, framecnt_t src_offset = 0);
+       void silence (samplecnt_t nframes, samplecnt_t offset = 0);
+       void read_from (const Buffer& src, samplecnt_t nframes, sampleoffset_t dst_offset = 0, sampleoffset_t src_offset = 0);
+       void merge_from (const Buffer& src, samplecnt_t nframes, sampleoffset_t dst_offset = 0, sampleoffset_t src_offset = 0);
 
        void copy(const MidiBuffer& copy);
+       void copy(MidiBuffer const * const);
+
+       void skip_to (TimeType when);
 
-       bool     push_back(const Evoral::MIDIEvent<TimeType>& event);
+       bool     push_back(const Evoral::Event<TimeType>& event);
        bool     push_back(TimeType time, size_t size, const uint8_t* data);
+
        uint8_t* reserve(TimeType time, size_t size);
 
        void resize(size_t);
        size_t size() const { return _size; }
        bool empty() const { return _size == 0; }
 
+       bool insert_event(const Evoral::Event<TimeType>& event);
        bool merge_in_place(const MidiBuffer &other);
 
+       /** EventSink interface for non-RT use (export, bounce). */
+       uint32_t write(TimeType time, Evoral::EventType type, uint32_t size, const uint8_t* buf);
+
        template<typename BufferType, typename EventType>
-       class iterator_base {
+               class iterator_base
+       {
        public:
-               iterator_base<BufferType, EventType>(BufferType& b, framecnt_t o) 
-                    : buffer(&b), offset(o) {}
-                iterator_base<BufferType, EventType>(const iterator_base<BufferType,EventType>& o) 
-                    : buffer (o.buffer), offset(o.offset) {}
-           
-               inline iterator_base<BufferType,EventType> operator= (const iterator_base<BufferType,EventType>& o) {
+               iterator_base<BufferType, EventType>(BufferType& b, samplecnt_t o)
+                       : buffer(&b), offset(o) {}
+
+               iterator_base<BufferType, EventType>(const iterator_base<BufferType,EventType>& o)
+                       : buffer (o.buffer), offset(o.offset) {}
+
+               inline iterator_base<BufferType,EventType> operator= (const iterator_base<BufferType,EventType>& o) {
                        if (&o != this) {
                                buffer = o.buffer;
                                offset = o.offset;
@@ -73,19 +87,24 @@ public:
                        uint8_t* ev_start = buffer->_data + offset + sizeof(TimeType);
                        int event_size = Evoral::midi_event_size(ev_start);
                        assert(event_size >= 0);
-                       return EventType(EventTypeMap::instance().midi_event_type(*ev_start),
+                       return EventType(midi_parameter_type(*ev_start),
                                        *((TimeType*)(buffer->_data + offset)),
                                        event_size, ev_start);
                }
+
                inline EventType operator*() {
                        uint8_t* ev_start = buffer->_data + offset + sizeof(TimeType);
                        int event_size = Evoral::midi_event_size(ev_start);
                        assert(event_size >= 0);
-                       return EventType(EventTypeMap::instance().midi_event_type(*ev_start),
-                                       *((TimeType*)(buffer->_data + offset)),
+                       return EventType(Evoral::MIDI_EVENT,
+                                       *(reinterpret_cast<TimeType*>((uintptr_t)(buffer->_data + offset))),
                                        event_size, ev_start);
                }
 
+               inline TimeType * timeptr() {
+                       return reinterpret_cast<TimeType*>((uintptr_t)(buffer->_data + offset));
+               }
+
                inline iterator_base<BufferType, EventType>& operator++() {
                        uint8_t* ev_start = buffer->_data + offset + sizeof(TimeType);
                        int event_size = Evoral::midi_event_size(ev_start);
@@ -93,18 +112,21 @@ public:
                        offset += sizeof(TimeType) + event_size;
                        return *this;
                }
+
                inline bool operator!=(const iterator_base<BufferType, EventType>& other) const {
                        return (buffer != other.buffer) || (offset != other.offset);
                }
+
                inline bool operator==(const iterator_base<BufferType, EventType>& other) const {
                        return (buffer == other.buffer) && (offset == other.offset);
                }
+
                BufferType*     buffer;
                size_t          offset;
        };
 
-       typedef iterator_base< MidiBuffer, Evoral::MIDIEvent<TimeType> >             iterator;
-       typedef iterator_base< const MidiBuffer, const Evoral::MIDIEvent<TimeType> > const_iterator;
+       typedef iterator_base< MidiBuffer, Evoral::Event<TimeType> >             iterator;
+       typedef iterator_base< const MidiBuffer, const Evoral::Event<TimeType> > const_iterator;
 
        iterator begin() { return iterator(*this, 0); }
        iterator end()   { return iterator(*this, _size); }
@@ -112,7 +134,7 @@ public:
        const_iterator begin() const { return const_iterator(*this, 0); }
        const_iterator end()   const { return const_iterator(*this, _size); }
 
-        iterator erase(const iterator& i) {
+       iterator erase(const iterator& i) {
                assert (i.buffer == this);
                uint8_t* ev_start = _data + i.offset + sizeof (TimeType);
                int event_size = Evoral::midi_event_size (ev_start);
@@ -147,24 +169,21 @@ public:
                return iterator (*this, i.offset);
        }
 
-       uint8_t* data() const { return _data; }
-
        /**
         * returns true if the message with the second argument as its MIDI
         * status byte should preceed the message with the first argument as
         * its MIDI status byte.
         */
        static bool second_simultaneous_midi_byte_is_first (uint8_t, uint8_t);
-       
+
 private:
-       friend class iterator_base< MidiBuffer, Evoral::MIDIEvent<TimeType> >;
-       friend class iterator_base< const MidiBuffer, const Evoral::MIDIEvent<TimeType> >;
+       friend class iterator_base< MidiBuffer, Evoral::Event<TimeType> >;
+       friend class iterator_base< const MidiBuffer, const Evoral::Event<TimeType> >;
 
        uint8_t* _data; ///< timestamp, event, timestamp, event, ...
        pframes_t _size;
 };
 
-
 } // namespace ARDOUR
 
 #endif // __ardour_midi_buffer_h__