| 
							- /*
 -  * Carla Native Plugins
 -  * Copyright (C) 2012-2015 Filipe Coelho <falktx@falktx.com>
 -  *
 -  * 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 the Free Software Foundation; either version 2 of
 -  * the License, or any later version.
 -  *
 -  * This program is distributed in the hope that it will be useful,
 -  * but WITHOUT ANY WARRANTY; without even the implied warranty of
 -  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 -  * GNU General Public License for more details.
 -  *
 -  * For a full copy of the GNU General Public License see the doc/GPL.txt file.
 -  */
 - 
 - #include "CarlaNativeExtUI.hpp"
 - #include "RtLinkedList.hpp"
 - 
 - #include "midi-base.hpp"
 - 
 - // -----------------------------------------------------------------------
 - 
 - class MidiSequencerPlugin : public NativePluginAndUiClass,
 -                             public AbstractMidiPlayer
 - {
 - public:
 -     enum Parameters {
 -         kParameterCount = 0
 -     };
 - 
 -     MidiSequencerPlugin(const NativeHostDescriptor* const host)
 -         : NativePluginAndUiClass(host, "midiseq-ui"),
 -           fNeedsAllNotesOff(false),
 -           fWantInEvents(false),
 -           fWasPlayingBefore(false),
 -           fTicksPerFrame(0.0),
 -           fInEvents(),
 -           fMidiOut(this),
 -           fTimeInfo(),
 -           leakDetector_MidiSequencerPlugin()
 -     {
 -         carla_zeroStruct(fTimeInfo);
 -     }
 - 
 - protected:
 -     // -------------------------------------------------------------------
 -     // Plugin process calls
 - 
 -     void process(float**, float**, const uint32_t frames, const NativeMidiEvent* const midiEvents, const uint32_t midiEventCount) override
 -     {
 -         if (const NativeTimeInfo* const timeInfo = getTimeInfo())
 -             fTimeInfo = *timeInfo;
 - 
 -         if (fWantInEvents)
 -         {
 -             if (midiEventCount > 0)
 -             {
 -                 RawMidiEvent rawMidiEvent;
 - 
 -                 for (uint32_t i=0; i < midiEventCount; ++i)
 -                 {
 -                     const NativeMidiEvent* const midiEvent(&midiEvents[i]);
 - 
 -                     rawMidiEvent.time    = fTimeInfo.playing ? fTimeInfo.frame + midiEvent->time : 0;
 -                     rawMidiEvent.size    = midiEvent->size;
 -                     rawMidiEvent.data[0] = midiEvent->data[0];
 -                     rawMidiEvent.data[1] = midiEvent->data[1];
 -                     rawMidiEvent.data[2] = midiEvent->data[2];
 -                     rawMidiEvent.data[3] = midiEvent->data[3];
 - 
 -                     fInEvents.appendRT(rawMidiEvent);
 -                 }
 -             }
 - 
 -             fInEvents.trySplice();
 -         }
 - 
 -         if (fWasPlayingBefore != fTimeInfo.playing)
 -         {
 -             fNeedsAllNotesOff = true;
 -             fWasPlayingBefore = fTimeInfo.playing;
 -         }
 - 
 -         if (fNeedsAllNotesOff)
 -         {
 -             NativeMidiEvent midiEvent;
 - 
 -             midiEvent.port    = 0;
 -             midiEvent.time    = 0;
 -             midiEvent.data[0] = 0;
 -             midiEvent.data[1] = MIDI_CONTROL_ALL_NOTES_OFF;
 -             midiEvent.data[2] = 0;
 -             midiEvent.data[3] = 0;
 -             midiEvent.size    = 3;
 - 
 -             for (int channel=MAX_MIDI_CHANNELS; --channel >= 0;)
 -             {
 -                 midiEvent.data[0] = uint8_t(MIDI_STATUS_CONTROL_CHANGE | (channel & MIDI_CHANNEL_BIT));
 -                 NativePluginAndUiClass::writeMidiEvent(&midiEvent);
 -             }
 - 
 -             fNeedsAllNotesOff = false;
 -         }
 - 
 -         if (fTimeInfo.playing)
 -         {
 -             if (! fTimeInfo.bbt.valid)
 -                 fTimeInfo.bbt.beatsPerMinute = 120.0;
 - 
 -             fTicksPerFrame = 48.0 / (60.0 / fTimeInfo.bbt.beatsPerMinute * getSampleRate());
 - 
 -             fMidiOut.play(fTicksPerFrame*static_cast<long double>(fTimeInfo.frame),
 -                           fTicksPerFrame*static_cast<double>(frames));
 -         }
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // Plugin UI calls
 - 
 -     void uiShow(const bool show) override
 -     {
 -         NativePluginAndUiClass::uiShow(show);
 - 
 -         if (show)
 -             _sendEventsToUI();
 -     }
 - 
 -     void uiIdle() override
 -     {
 -         NativePluginAndUiClass::uiIdle();
 - 
 -         // send transport
 -         if (isPipeRunning())
 -         {
 -             char strBuf[0xff+1];
 -             strBuf[0xff] = '\0';
 - 
 -             const float  beatsPerBar    = fTimeInfo.bbt.valid ? fTimeInfo.bbt.beatsPerBar    : 4.0f;
 -             const double beatsPerMinute = fTimeInfo.bbt.valid ? fTimeInfo.bbt.beatsPerMinute : 120.0;
 -             const float  beatType       = fTimeInfo.bbt.valid ? fTimeInfo.bbt.beatType       : 4.0f;
 - 
 -             const double ticksPerBeat  = 48.0;
 -             const double ticksPerFrame = ticksPerBeat / (60.0 / beatsPerMinute * getSampleRate());
 -             const double fullTicks     = static_cast<double>(ticksPerFrame*static_cast<long double>(fTimeInfo.frame));
 -             const double fullBeats     = fullTicks/ticksPerBeat;
 - 
 -             const uint32_t tick = static_cast<uint32_t>(std::floor(std::fmod(fullTicks, ticksPerBeat)));
 -             const uint32_t beat = static_cast<uint32_t>(std::floor(std::fmod(fullBeats, static_cast<double>(beatsPerBar))));
 -             const uint32_t bar  = static_cast<uint32_t>(std::floor(fullBeats/beatsPerBar));
 - 
 -             const CarlaMutexLocker cml(getPipeLock());
 -             const ScopedLocale csl;
 - 
 -             writeAndFixMessage("transport");
 -             writeMessage(fTimeInfo.playing ? "true\n" : "false\n");
 - 
 -             std::sprintf(strBuf, P_UINT64 ":%i:%i:%i\n", fTimeInfo.frame, bar, beat, tick);
 -             writeMessage(strBuf);
 - 
 -             std::sprintf(strBuf, "%f:%f:%f\n", beatsPerMinute, beatsPerBar, beatType);
 -             writeMessage(strBuf);
 - 
 -             flushMessages();
 -         }
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // Plugin state calls
 - 
 -     char* getState() const override
 -     {
 -         return fMidiOut.getState();
 -     }
 - 
 -     void setState(const char* const data) override
 -     {
 -         fMidiOut.setState(data);
 - 
 -         if (isPipeRunning())
 -             _sendEventsToUI();
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // AbstractMidiPlayer calls
 - 
 -     void writeMidiEvent(const uint8_t port, const long double timePosFrame, const RawMidiEvent* const event) override
 -     {
 -         NativeMidiEvent midiEvent;
 - 
 -         midiEvent.port    = port;
 -         midiEvent.time    = uint32_t(timePosFrame/fTicksPerFrame);
 -         midiEvent.data[0] = event->data[0];
 -         midiEvent.data[1] = event->data[1];
 -         midiEvent.data[2] = event->data[2];
 -         midiEvent.data[3] = event->data[3];
 -         midiEvent.size    = event->size;
 - 
 -         carla_stdout("Playing at %i :: %03X:%03i:%03i",
 -                      midiEvent.time, midiEvent.data[0], midiEvent.data[1], midiEvent.data[2]);
 - 
 -         NativePluginAndUiClass::writeMidiEvent(&midiEvent);
 -     }
 - 
 -     // -------------------------------------------------------------------
 -     // Pipe Server calls
 - 
 -     bool msgReceived(const char* const msg) noexcept override
 -     {
 -         if (NativePluginAndUiClass::msgReceived(msg))
 -             return true;
 - 
 -         if (std::strcmp(msg, "midi-clear-all") == 0)
 -         {
 -             fMidiOut.clear();
 -             fNeedsAllNotesOff = true;
 -             return true;
 -         }
 - 
 -         if (std::strcmp(msg, "midievent-add") == 0)
 -         {
 -             uint64_t time;
 -             uint8_t size;
 - 
 -             CARLA_SAFE_ASSERT_RETURN(readNextLineAsULong(time), true);
 -             CARLA_SAFE_ASSERT_RETURN(readNextLineAsByte(size), true);
 - 
 -             uint8_t data[size], dvalue;
 - 
 -             for (uint8_t i=0; i<size; ++i)
 -             {
 -                 CARLA_SAFE_ASSERT_RETURN(readNextLineAsByte(dvalue), true);
 -                 data[i] = dvalue;
 -             }
 - 
 -             fMidiOut.addRaw(time, data, size);
 - 
 -             return true;
 -         }
 - 
 -         if (std::strcmp(msg, "midievent-remove") == 0)
 -         {
 -             uint64_t time;
 -             uint8_t size;
 - 
 -             CARLA_SAFE_ASSERT_RETURN(readNextLineAsULong(time), true);
 -             CARLA_SAFE_ASSERT_RETURN(readNextLineAsByte(size), true);
 - 
 -             uint8_t data[size], dvalue;
 - 
 -             for (uint8_t i=0; i<size; ++i)
 -             {
 -                 CARLA_SAFE_ASSERT_RETURN(readNextLineAsByte(dvalue), true);
 -                 data[i] = dvalue;
 -             }
 - 
 -             fMidiOut.removeRaw(time, data, size);
 - 
 -             return true;
 -         }
 - 
 -         return false;
 -     }
 - 
 -     // -------------------------------------------------------------------
 - 
 - private:
 -     bool fNeedsAllNotesOff;
 -     bool fWantInEvents;
 -     bool fWasPlayingBefore;
 - 
 -     double fTicksPerFrame;
 - 
 -     struct InRtEvents {
 -         CarlaMutex mutex;
 -         RtLinkedList<RawMidiEvent>::Pool dataPool;
 -         RtLinkedList<RawMidiEvent> data;
 -         RtLinkedList<RawMidiEvent> dataPendingRT;
 - 
 -         InRtEvents() noexcept
 -             : mutex(),
 -               dataPool(MIN_PREALLOCATED_EVENT_COUNT, MAX_PREALLOCATED_EVENT_COUNT),
 -               data(dataPool),
 -               dataPendingRT(dataPool) {}
 - 
 -         ~InRtEvents() noexcept
 -         {
 -             clear();
 -         }
 - 
 -         void appendRT(const RawMidiEvent& event) noexcept
 -         {
 -             dataPendingRT.append(event);
 -         }
 - 
 -         void clear() noexcept
 -         {
 -             mutex.lock();
 -             data.clear();
 -             dataPendingRT.clear();
 -             mutex.unlock();
 -         }
 - 
 -         void trySplice() noexcept
 -         {
 -             if (mutex.tryLock())
 -             {
 -                 if (dataPendingRT.count() > 0)
 -                     dataPendingRT.moveTo(data, true);
 -                 mutex.unlock();
 -             }
 -         }
 - 
 -         CARLA_DECLARE_NON_COPY_STRUCT(InRtEvents);
 - 
 -     } fInEvents;
 - 
 -     MidiPattern    fMidiOut;
 -     NativeTimeInfo fTimeInfo;
 - 
 -     void _sendEventsToUI() const noexcept
 -     {
 -         char strBuf[0xff+1];
 -         strBuf[0xff] = '\0';
 - 
 -         const CarlaMutexLocker cml1(getPipeLock());
 -         const CarlaMutexLocker cml2(fMidiOut.getLock());
 - 
 -         writeMessage("midi-clear-all\n", 15);
 - 
 -         for (LinkedList<const RawMidiEvent*>::Itenerator it = fMidiOut.iteneratorBegin(); it.valid(); it.next())
 -         {
 -             const RawMidiEvent* const rawMidiEvent(it.getValue(nullptr));
 -             CARLA_SAFE_ASSERT_CONTINUE(rawMidiEvent != nullptr);
 - 
 -             writeMessage("midievent-add\n", 14);
 - 
 -             std::snprintf(strBuf, 0xff, P_INT64 "\n", rawMidiEvent->time);
 -             writeMessage(strBuf);
 - 
 -             std::snprintf(strBuf, 0xff, "%i\n", rawMidiEvent->size);
 -             writeMessage(strBuf);
 - 
 -             for (uint8_t i=0, size=rawMidiEvent->size; i<size; ++i)
 -             {
 -                 std::snprintf(strBuf, 0xff, "%i\n", rawMidiEvent->data[i]);
 -                 writeMessage(strBuf);
 -             }
 -         }
 -     }
 - 
 -     PluginClassEND(MidiSequencerPlugin)
 -     CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(MidiSequencerPlugin)
 - };
 - 
 - // -----------------------------------------------------------------------
 - 
 - static const NativePluginDescriptor midisequencerDesc = {
 -     /* category  */ NATIVE_PLUGIN_CATEGORY_UTILITY,
 -     /* hints     */ static_cast<NativePluginHints>(NATIVE_PLUGIN_IS_RTSAFE
 -                                                   |NATIVE_PLUGIN_HAS_UI
 -                                                   |NATIVE_PLUGIN_USES_STATE
 -                                                   |NATIVE_PLUGIN_USES_TIME),
 -     /* supports  */ NATIVE_PLUGIN_SUPPORTS_EVERYTHING,
 -     /* audioIns  */ 0,
 -     /* audioOuts */ 0,
 -     /* midiIns   */ 1,
 -     /* midiOuts  */ 1,
 -     /* paramIns  */ MidiSequencerPlugin::kParameterCount,
 -     /* paramOuts */ 0,
 -     /* name      */ "MIDI Sequencer",
 -     /* label     */ "midisequencer",
 -     /* maker     */ "falkTX, tatch",
 -     /* copyright */ "GNU GPL v2+",
 -     PluginDescriptorFILL(MidiSequencerPlugin)
 - };
 - 
 - // -----------------------------------------------------------------------
 - 
 - CARLA_EXPORT
 - void carla_register_native_plugin_midisequencer();
 - 
 - CARLA_EXPORT
 - void carla_register_native_plugin_midisequencer()
 - {
 -     carla_register_native_plugin(&midisequencerDesc);
 - }
 - 
 - // -----------------------------------------------------------------------
 
 
  |