|  | /*
 * Carla Backend
 * Copyright (C) 2012 Filipe Coelho <falktx@gmail.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 COPYING file
 */
#ifndef CARLA_ENGINE_H
#define CARLA_ENGINE_H
#include "carla_osc.h"
#include "carla_shared.h"
#include "carla_threads.h"
#ifdef CARLA_ENGINE_JACK
#include <jack/jack.h>
#include <jack/midiport.h>
#endif
#ifdef CARLA_ENGINE_RTAUDIO
#include "RtAudio.h"
#include "RtMidi.h"
#endif
CARLA_BACKEND_START_NAMESPACE
/*!
 * @defgroup CarlaBackendEngine Carla Backend Engine
 *
 * The Carla Backend Engine
 * @{
 */
/*!
 * @defgroup TimeInfoValidHints TimeInfo Valid Hints
 *
 * Various hints used for CarlaTimeInfo::valid.
 * @{
 */
const uint32_t CarlaEngineTimeBBT = 0x1;
/**@}*/
enum CarlaEngineType {
    CarlaEngineTypeNull,
    CarlaEngineTypeJack,
    CarlaEngineTypeRtAudio
};
enum CarlaEnginePortType {
    CarlaEnginePortTypeAudio,
    CarlaEnginePortTypeControl,
    CarlaEnginePortTypeMIDI
};
enum CarlaEngineControlEventType {
    CarlaEngineEventNull = 0,
    CarlaEngineEventControlChange,
    CarlaEngineEventMidiBankChange,
    CarlaEngineEventMidiProgramChange,
    CarlaEngineEventAllSoundOff,
    CarlaEngineEventAllNotesOff
};
struct CarlaEngineControlEvent {
    CarlaEngineControlEventType type;
    uint32_t time;
    uint8_t  channel;
    uint16_t controller;
    double   value;
    CarlaEngineControlEvent()
        : type(CarlaEngineEventNull),
          time(0),
          channel(0),
          controller(0),
          value(0.0) {}
};
struct CarlaEngineMidiEvent {
    uint32_t time;
    uint8_t  size;
    uint8_t  data[4];
    CarlaEngineMidiEvent()
        : time(0),
      #ifdef Q_COMPILER_INITIALIZER_LISTS
          size(0),
          data{0} {}
      #else
          size(0) { memset(data, 0, sizeof(uint8_t)*4); }
      #endif
};
struct CarlaTimeInfo {
    bool playing;
    uint32_t frame;
    uint32_t time;
    uint32_t valid;
    struct {
        int32_t bar;
        int32_t beat;
        int32_t tick;
        double bar_start_tick;
        float  beats_per_bar;
        float  beat_type;
        double ticks_per_beat;
        double beats_per_minute;
    } bbt;
    CarlaTimeInfo()
        : playing(false),
          frame(0),
          time(0),
      #ifdef Q_COMPILER_INITIALIZER_LISTS
          valid(0),
          bbt{0, 0, 0, 0.0, 0.0f, 0.0f, 0.0, 0.0} {}
      #else
          valid(0)
    {
        bbt.bar = bbt.beat = bbt.tick = 0;
        bbt.beats_per_bar  = bbt.beat_type = 0.0f;
        bbt.bar_start_tick = bbt.ticks_per_beat = bbt.beats_per_minute = 0.0;
    }
      #endif
};
struct CarlaEngineClientNativeHandle {
#ifdef CARLA_ENGINE_JACK
    jack_client_t* jackClient;
#endif
#ifdef CARLA_ENGINE_RTAUDIO
    RtAudio* rtAudioPtr;
#endif
    CarlaEngineClientNativeHandle()
    {
#ifdef CARLA_ENGINE_JACK
        jackClient = nullptr;
#endif
#ifdef CARLA_ENGINE_RTAUDIO
        rtAudioPtr = nullptr;
#endif
    }
};
struct CarlaEnginePortNativeHandle {
#ifdef CARLA_ENGINE_JACK
    jack_client_t* jackClient;
    jack_port_t* jackPort;
#endif
    CarlaEnginePortNativeHandle()
    {
#ifdef CARLA_ENGINE_JACK
        jackClient = nullptr;
        jackPort = nullptr;
#endif
    }
};
// -----------------------------------------------------------------------
class CarlaEngineClient;
class CarlaEngineBasePort;
/*!
 * \class CarlaEngine
 *
 * \brief Carla Backend base engine class
 *
 * This is the base class for all available engine types available in Carla Backend.
 */
class CarlaEngine
{
public:
    CarlaEngine();
    virtual ~CarlaEngine();
    // -------------------------------------------------------------------
    // Static values
    static const unsigned short MAX_PEAKS = 2;
    static int maxClientNameSize();
    static int maxPortNameSize();
    // -------------------------------------------------------------------
    // Virtual, per-engine type calls
    virtual bool init(const char* const clientName);
    virtual bool close();
    virtual bool isOnAudioThread() = 0;
    virtual bool isOffline() = 0;
    virtual bool isRunning() = 0;
    virtual CarlaEngineClient* addClient(CarlaPlugin* const plugin) = 0;
    // -------------------------------------------------------------------
    // Plugin management
    short        getNewPluginId() const;
    CarlaPlugin* getPlugin(const unsigned short id) const;
    CarlaPlugin* getPluginUnchecked(const unsigned short id) const;
    const char*  getUniqueName(const char* const name);
    short addPlugin(const BinaryType btype, const PluginType ptype, const char* const filename, const char* const name, const char* const label, void* const extra = nullptr);
    short addPlugin(const PluginType ptype, const char* const filename, const char* const name, const char* const label, void* const extra = nullptr);
    bool  removePlugin(const unsigned short id);
    void  removeAllPlugins();
    void  idlePluginGuis();
    // bridge, internal use only
    void __bridgePluginRegister(const unsigned short id, CarlaPlugin* const plugin)
    {
        m_carlaPlugins[id] = plugin;
    }
    // -------------------------------------------------------------------
    // Information (base)
    const char* getName() const;
    double   getSampleRate() const;
    uint32_t getBufferSize() const;
    const CarlaTimeInfo* getTimeInfo() const;
    // -------------------------------------------------------------------
    // Information (audio peaks)
    double getInputPeak(const unsigned short pluginId, const unsigned short id) const;
    double getOutputPeak(const unsigned short pluginId, const unsigned short id) const;
    void   setInputPeak(const unsigned short pluginId, const unsigned short id, double value);
    void   setOutputPeak(const unsigned short pluginId, const unsigned short id, double value);
    // -------------------------------------------------------------------
    // Callback
    void callback(const CallbackType action, const unsigned short pluginId, const int value1, const int value2, const double value3);
    void setCallback(const CallbackFunc func, void* const ptr);
    // -------------------------------------------------------------------
    // Mutex locks
    void processLock();
    void processUnlock();
    void midiLock();
    void midiUnlock();
#ifndef BUILD_BRIDGE
    // -------------------------------------------------------------------
    // OSC Stuff
    bool isOscControllerRegisted() const;
    const char* getOscServerPath() const;
    void osc_send_add_plugin(const int32_t pluginId, const char* const pluginName);
    void osc_send_remove_plugin(const int32_t pluginId);
    void osc_send_set_plugin_data(const int32_t pluginId, const int32_t type, const int32_t category, const int32_t hints, const char* const realName, const char* const label, const char* const maker, const char* const copyright, const int64_t uniqueId);
    void osc_send_set_plugin_ports(const int32_t pluginId, const int32_t audioIns, const int32_t audioOuts, const int32_t midiIns, const int32_t midiOuts, const int32_t cIns, const int32_t cOuts, const int32_t cTotals);
    void osc_send_set_parameter_data(const int32_t pluginId, const int32_t index, const int32_t type, const int32_t hints, const char* const name, const char* const label, const double current);
    void osc_send_set_parameter_ranges(const int32_t pluginId, const int32_t index, const double min, const double max, const double def, const double step, const double stepSmall, const double stepLarge);
    void osc_send_set_parameter_midi_cc(const int32_t pluginId, const int32_t index, const int32_t cc);
    void osc_send_set_parameter_midi_channel(const int32_t pluginId, const int32_t index, const int32_t channel);
    void osc_send_set_parameter_value(const int32_t pluginId, const int32_t index, const double value);
    void osc_send_set_default_value(const int32_t pluginId, const int32_t index, const double value);
    void osc_send_set_program(const int32_t pluginId, const int32_t index);
    void osc_send_set_program_count(const int32_t pluginId, const int32_t count);
    void osc_send_set_program_name(const int32_t pluginId, const int32_t index, const char* const name);
    void osc_send_set_midi_program(const int32_t pluginId, const int32_t index);
    void osc_send_set_midi_program_count(const int32_t pluginId, const int32_t count);
    void osc_send_set_midi_program_data(const int32_t pluginId, const int32_t index, const int32_t bank, const int32_t program, const char* const name);
    void osc_send_set_input_peak_value(const int32_t pluginId, const int32_t portId, const double value);
    void osc_send_set_output_peak_value(const int32_t pluginId, const int32_t portId, const double value);
    void osc_send_note_on(const int32_t pluginId, const int32_t channel, const int32_t note, const int32_t velo);
    void osc_send_note_off(const int32_t pluginId, const int32_t channel, const int32_t note);
    void osc_send_exit();
    // -------------------------------------------------------------------
    // Rack mode
    static const unsigned short MAX_ENGINE_CONTROL_EVENTS = 512;
    static const unsigned short MAX_ENGINE_MIDI_EVENTS    = 512;
    CarlaEngineControlEvent rackControlEventsIn[MAX_ENGINE_CONTROL_EVENTS];
    CarlaEngineControlEvent rackControlEventsOut[MAX_ENGINE_CONTROL_EVENTS];
    CarlaEngineMidiEvent rackMidiEventsIn[MAX_ENGINE_MIDI_EVENTS];
    CarlaEngineMidiEvent rackMidiEventsOut[MAX_ENGINE_MIDI_EVENTS];
#endif
    // -------------------------------------
    /*!
     * \class ScopedLocker
     *
     * \brief Carla engine scoped locker
     *
     * This is a handy class that temporarily locks an engine during a function scope.
     */
    class ScopedLocker
    {
    public:
        /*!
         * Lock the engine \a engine if \a lock is true.
         * The engine is unlocked in the deconstructor of this class if \a lock is true.
         *
         * \param engine The engine to lock
         * \param lock Wherever to lock the engine or not, true by default
         */
        ScopedLocker(CarlaEngine* const engine, bool lock = true) :
            m_engine(engine),
            m_lock(lock)
        {
            if (m_lock)
                m_engine->processLock();
        }
        ~ScopedLocker()
        {
            if (m_lock)
                m_engine->processUnlock();
        }
    private:
        CarlaEngine* const m_engine;
        const bool m_lock;
    };
    // -------------------------------------
protected:
    CarlaEngineType type;
    const char* name;
    uint32_t bufferSize;
    double   sampleRate;
    CarlaTimeInfo timeInfo;
    unsigned short maxPluginNumber;
    void bufferSizeChanged(uint32_t newBufferSize);
private:
    CarlaCheckThread m_checkThread;
#ifndef BUILD_BRIDGE
    CarlaOsc m_osc;
    const CarlaOscData* m_oscData;
#endif
    QMutex m_procLock;
    QMutex m_midiLock;
    CallbackFunc m_callback;
    void*        m_callbackPtr;
    CarlaPlugin* m_carlaPlugins[MAX_PLUGINS];
    const char*  m_uniqueNames[MAX_PLUGINS];
    double m_insPeak[MAX_PLUGINS * MAX_PEAKS];
    double m_outsPeak[MAX_PLUGINS * MAX_PEAKS];
};
// -----------------------------------------------------------------------
class CarlaEngineClient
{
public:
    CarlaEngineClient(const CarlaEngineType& type, const CarlaEngineClientNativeHandle& handle);
    ~CarlaEngineClient();
    void activate();
    void deactivate();
    bool isActive() const;
    bool isOk() const;
    const CarlaEngineBasePort* addPort(const CarlaEnginePortType type, const char* const name, const bool isInput);
private:
    bool m_active;
    const CarlaEngineType type;
    const CarlaEngineClientNativeHandle handle;
};
// -----------------------------------------------------------------------
class CarlaEngineBasePort
{
public:
    CarlaEngineBasePort(const CarlaEnginePortNativeHandle& handle, const bool isInput);
    virtual ~CarlaEngineBasePort();
    virtual void initBuffer(CarlaEngine* const engine) = 0;
protected:
    void* buffer;
    const bool isInput;
    const CarlaEnginePortNativeHandle handle;
};
class CarlaEngineAudioPort : public CarlaEngineBasePort
{
public:
    CarlaEngineAudioPort(const CarlaEnginePortNativeHandle& handle, const bool isInput);
    void initBuffer(CarlaEngine* const engine);
#ifdef CARLA_ENGINE_JACK
    float* getJackAudioBuffer(uint32_t nframes);
#endif
};
class CarlaEngineControlPort : public CarlaEngineBasePort
{
public:
    CarlaEngineControlPort(const CarlaEnginePortNativeHandle& handle, const bool isInput);
    void initBuffer(CarlaEngine* const engine);
    uint32_t getEventCount();
    const CarlaEngineControlEvent* getEvent(uint32_t index);
    void writeEvent(CarlaEngineControlEventType type, uint32_t time, uint8_t channel, uint8_t controller, double value);
};
class CarlaEngineMidiPort : public CarlaEngineBasePort
{
public:
    CarlaEngineMidiPort(const CarlaEnginePortNativeHandle& handle, const bool isInput);
    void initBuffer(CarlaEngine* const engine);
    uint32_t getEventCount();
    const CarlaEngineMidiEvent* getEvent(uint32_t index);
    void writeEvent(uint32_t time, const uint8_t* data, uint8_t size);
};
// -----------------------------------------------------------------------
#ifdef CARLA_ENGINE_JACK
class CarlaEngineJack : public CarlaEngine
{
public:
    CarlaEngineJack();
    ~CarlaEngineJack();
    // -------------------------------------
    bool init(const char* const clientName);
    bool close();
    bool isOnAudioThread();
    bool isOffline();
    bool isRunning();
    CarlaEngineClient* addClient(CarlaPlugin* const plugin);
    // -------------------------------------
    void handleSampleRateCallback(double newSampleRate);
    void handleBufferSizeCallback(uint32_t newBufferSize);
    void handleFreewheelCallback(bool isFreewheel);
    void handleProcessCallback(uint32_t nframes);
    void handleShutdownCallback();
    // -------------------------------------
private:
    jack_client_t* client;
    jack_transport_state_t state;
    jack_position_t pos;
    bool freewheel;
    QThread* procThread;
    // -------------------------------------
    static const unsigned short rackPortAudioIn1   = 0;
    static const unsigned short rackPortAudioIn2   = 1;
    static const unsigned short rackPortAudioOut1  = 2;
    static const unsigned short rackPortAudioOut2  = 3;
    static const unsigned short rackPortControlIn  = 4;
    static const unsigned short rackPortControlOut = 5;
    static const unsigned short rackPortMidiIn     = 6;
    static const unsigned short rackPortMidiOut    = 7;
    static const unsigned short rackPortCount      = 8;
    jack_port_t* rackJackPorts[rackPortCount];
};
#endif
// -----------------------------------------------------------------------
#ifdef CARLA_ENGINE_RTAUDIO
class CarlaEngineRtAudio : public CarlaEngine
{
public:
    CarlaEngineRtAudio(RtAudio::Api api);
    ~CarlaEngineRtAudio();
    // -------------------------------------
    bool init(const char* const clientName);
    bool close();
    bool isOnAudioThread();
    bool isOffline();
    bool isRunning();
    CarlaEngineClient* addClient(CarlaPlugin* const plugin);
    // -------------------------------------
    void handleProcessCallback(void* outputBuffer, void* inputBuffer, unsigned int nframes, double streamTime, RtAudioStreamStatus status);
private:
    RtAudio adac;
    QThread* procThread;
};
#endif
/**@}*/
CARLA_BACKEND_END_NAMESPACE
#endif // CARLA_ENGINE_H
 |