|  | /*
 * Carla Engine API
 * Copyright (C) 2012-2013 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 GPL.txt file
 */
#ifndef __CARLA_ENGINE_HPP__
#define __CARLA_ENGINE_HPP__
#include "carla_backend.hpp"
#include "carla_utils.hpp"
#ifdef BUILD_BRIDGE
struct CarlaOscData;
#else
class QProcessEnvironment;
#endif
CARLA_BACKEND_START_NAMESPACE
#if 0
} // Fix editor indentation
#endif
// -----------------------------------------------------------------------
/*!
 * The type of an engine.
 */
enum EngineType {
    /*!
    * Null engine type.
    */
    kEngineTypeNull = 0,
    /*!
    * JACK engine type.\n
    * Provides all processing modes.
    */
    kEngineTypeJack = 1,
    /*!
    * RtAudio engine type, used to provide Native Audio and MIDI support.
    */
    kEngineTypeRtAudio = 2,
    /*!
    * Plugin engine type, used to export the engine as a plugin.
    */
    kEngineTypePlugin = 3
};
/*!
 * The type of an engine port.
 */
enum EnginePortType {
    /*!
    * Null port type.
    */
    kEnginePortTypeNull = 0,
    /*!
    * Audio port type.
    * \see CarlaEngineAudioPort
    */
    kEnginePortTypeAudio = 1,
    /*!
    * Event port type.
    ** \see CarlaEngineEventPort
    */
    kEnginePortTypeEvent = 2
};
/*!
 * The type of an engine event.
 */
enum EngineEventType {
    /*!
    * Null port type.
    */
    kEngineEventTypeNull = 0,
    /*!
    * Control event type.
    * \see EngineControlEvent
    */
    kEngineEventTypeControl = 1,
    /*!
    * MIDI event type.
    * \see EngineMidiEvent
    */
    kEngineEventTypeMidi = 2
};
/*!
 * The type of an engine control event.
 */
enum EngineControlEventType {
    /*!
    * Null event type.
    */
    kEngineControlEventTypeNull = 0,
    /*!
    * Parameter event type.\n
    * \note Value uses a range of 0.0<->1.0.
    */
    kEngineControlEventTypeParameter = 1,
    /*!
    * MIDI Bank event type.
    */
    kEngineControlEventTypeMidiBank = 2,
    /*!
    * MIDI Program change event type.
    */
    kEngineControlEventTypeMidiProgram = 3,
    /*!
    * All sound off event type.
    */
    kEngineControlEventTypeAllSoundOff = 4,
    /*!
    * All notes off event type.
    */
    kEngineControlEventTypeAllNotesOff = 5
};
/*!
 * Engine control event.
 */
struct EngineControlEvent {
    EngineControlEventType type; //!< Control-Event type.
    uint16_t param;              //!< Parameter ID, midi bank or midi program.
    double   value;              //!< Parameter value, normalized to 0.0<->1.0.
    EngineControlEvent()
    {
        clear();
    }
    void clear()
    {
        type  = kEngineControlEventTypeNull;
        param = 0;
        value = 0.0;
    }
};
/*!
 * Engine MIDI event.
 */
struct EngineMidiEvent {
    uint8_t port;    //!< Port offset (usually 0)
    uint8_t data[3]; //!< MIDI data, without channel bit
    uint8_t size;    //!< Number of bytes used
    EngineMidiEvent()
    {
        clear();
    }
    void clear()
    {
        port    = 0;
        data[0] = 0;
        data[1] = 0;
        data[2] = 0;
        size    = 0;
    }
};
/*!
 * Engine event.
 */
struct EngineEvent {
    EngineEventType type; //!< Event Type; either Control or MIDI
    uint32_t time;        //!< Time offset in frames
    uint8_t  channel;     //!< Channel, used for MIDI-related events
    union {
        EngineControlEvent ctrl;
        EngineMidiEvent midi;
    };
    EngineEvent()
    {
        clear();
    }
    void clear()
    {
        type = kEngineEventTypeNull;
        time = 0;
        channel = 0;
    }
};
// -----------------------------------------------------------------------
/*!
 * Engine devices
 */
struct EngineDevices {
    struct Audio {
        uint32_t     count;
        int32_t      current;
        const char** names;
    } audio;
    struct Midi {
        uint32_t     count;
        const char** names;
        bool*        selected;
    } midi;
    struct Info {
        unsigned int* bufferSizes; // valid until 0 value reached
        unsigned int* sampleRates; // valid until 0 value reached
    } info;
    EngineDevices()
    {
        audio.count   = 0;
        audio.current = -1;
        audio.names   = nullptr;
        midi.count    = 0;
        midi.names    = nullptr;
        midi.selected = nullptr;
        info.bufferSizes = nullptr;
        info.sampleRates = nullptr;
    }
};
/*!
 * Engine options.
 */
struct EngineOptions {
    ProcessMode processMode;
    bool forceStereo;
    bool preferPluginBridges;
    bool preferUiBridges;
#ifdef WANT_DSSI
    bool useDssiVstChunks;
#endif
    unsigned int maxParameters;
    unsigned int oscUiTimeout;
    unsigned int preferredBufferSize;
    unsigned int preferredSampleRate;
#ifndef BUILD_BRIDGE
    CarlaString bridge_native;
    CarlaString bridge_posix32;
    CarlaString bridge_posix64;
    CarlaString bridge_win32;
    CarlaString bridge_win64;
#endif
#ifdef WANT_LV2
    CarlaString bridge_lv2gtk2;
    CarlaString bridge_lv2gtk3;
    CarlaString bridge_lv2qt4;
    CarlaString bridge_lv2qt5;
    CarlaString bridge_lv2cocoa;
    CarlaString bridge_lv2win;
    CarlaString bridge_lv2x11;
#endif
#ifdef WANT_VST
    CarlaString bridge_vstcocoa;
    CarlaString bridge_vsthwnd;
    CarlaString bridge_vstx11;
#endif
    EngineOptions()
        : processMode(PROCESS_MODE_CONTINUOUS_RACK),
          forceStereo(false),
          preferPluginBridges(false),
          preferUiBridges(true),
#ifdef WANT_DSSI
          useDssiVstChunks(false),
#endif
          maxParameters(MAX_DEFAULT_PARAMETERS),
          oscUiTimeout(4000),
          preferredBufferSize(512),
          preferredSampleRate(44100) {}
};
/*!
 * Engine BBT Time information.
 */
struct EngineTimeInfoBBT {
    int32_t bar;  //!< current bar
    int32_t beat; //!< current beat-within-bar
    int32_t tick; //!< current tick-within-beat
    double barStartTick;
    float beatsPerBar; //!< time signature "numerator"
    float beatType;    //!< time signature "denominator"
    double ticksPerBeat;
    double beatsPerMinute;
    EngineTimeInfoBBT()
        : bar(0),
          beat(0),
          tick(0),
          barStartTick(0.0),
          beatsPerBar(0.0f),
          beatType(0.0f),
          ticksPerBeat(0.0),
          beatsPerMinute(0.0) {}
};
/*!
 * Engine Time information.
 */
struct EngineTimeInfo {
    static const uint32_t ValidBBT = 0x1;
    bool playing;
    uint32_t frame;
    uint64_t time;
    uint32_t valid;
    EngineTimeInfoBBT bbt;
    EngineTimeInfo()
    {
        clear();
    }
    void clear()
    {
        playing = false;
        frame   = 0;
        time    = 0;
        valid   = 0x0;
    }
};
// -----------------------------------------------------------------------
/*!
 * Carla Engine port (Abstract).\n
 * This is the base class for all Carla Engine ports.
 */
class CarlaEnginePort
{
public:
    /*!
     * The contructor.\n
     * Param \a isInput defines wherever this is an input port or not (output otherwise).\n
     * Input/output state and process mode are constant for the lifetime of the port.
     */
    CarlaEnginePort(const bool isInput, const ProcessMode processMode);
    /*!
     * The destructor.
     */
    virtual ~CarlaEnginePort();
    /*!
     * Get the type of the port, as provided by the respective subclasses.
     */
    virtual EnginePortType type() const = 0;
    /*!
     * Initialize the port's internal buffer for \a engine.
     */
    virtual void initBuffer(CarlaEngine* const engine) = 0;
protected:
    const bool kIsInput;
    const ProcessMode kProcessMode;
private:
    CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEnginePort)
};
// -----------------------------------------------------------------------
/*!
 * Carla Engine Audio port.
 */
class CarlaEngineAudioPort : public CarlaEnginePort
{
public:
    /*!
     * The contructor.\n
     * All constructor parameters are constant and will never change in the lifetime of the port.
     */
    CarlaEngineAudioPort(const bool isInput, const ProcessMode processMode);
    /*!
     * The destructor.
     */
    virtual ~CarlaEngineAudioPort();
    /*!
     * Get the type of the port, in this case CarlaEnginePortTypeAudio.
     */
    EnginePortType type() const
    {
        return kEnginePortTypeAudio;
    }
    /*!
     * Initialize the port's internal buffer for \a engine.
     */
    virtual void initBuffer(CarlaEngine* const engine);
    /*!
     * Direct access to the port's audio buffer.
     */
    float* getBuffer() const
    {
        return fBuffer;
    }
protected:
    float* fBuffer;
private:
    CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineAudioPort)
};
// -----------------------------------------------------------------------
/*!
 * Carla Engine Event port.
 */
class CarlaEngineEventPort : public CarlaEnginePort
{
public:
    /*!
     * The contructor.\n
     * All constructor parameters are constant and will never change in the lifetime of the port.
     */
    CarlaEngineEventPort(const bool isInput, const ProcessMode processMode);
    /*!
     * The destructor.
     */
    virtual ~CarlaEngineEventPort();
    /*!
     * Get the type of the port, in this case CarlaEnginePortTypeControl.
     */
    EnginePortType type() const
    {
        return kEnginePortTypeEvent;
    }
    /*!
     * Initialize the port's internal buffer for \a engine.
     */
    virtual void initBuffer(CarlaEngine* const engine);
    /*!
     * Get the number of events present in the buffer.
     * \note You must only call this for input ports.
     */
    virtual uint32_t getEventCount();
    /*!
     * Get the event at \a index.
     * \note You must only call this for input ports.
     */
    virtual const EngineEvent& getEvent(const uint32_t index);
    /*!
     * Write a control event into the buffer.\n
     * Arguments are the same as in the EngineControlEvent struct.
     * \note You must only call this for output ports.
     */
    virtual void writeControlEvent(const uint32_t time, const uint8_t channel, const EngineControlEventType type, const uint16_t param, const double value = 0.0);
    /*!
     * Write a control event into the buffer, overloaded call.
     */
    void writeControlEvent(const uint32_t time, const uint8_t channel, const EngineControlEvent& ctrl)
    {
        writeControlEvent(time, channel, ctrl.type, ctrl.param, ctrl.value);
    }
    /*!
     * Write a MIDI event into the buffer.\n
     * Arguments are the same as in the EngineMidiEvent struct.
     ** \note You must only call this for output ports.
     */
    virtual void writeMidiEvent(const uint32_t time, const uint8_t channel, const uint8_t port, const uint8_t* const data, const uint8_t size);
    /*!
     * Write a MIDI event into the buffer, overloaded call.
     */
    void writeMidiEvent(const uint32_t time, const uint8_t channel, const EngineMidiEvent& midi)
    {
        writeMidiEvent(time, channel, midi.port, midi.data, midi.size);
    }
private:
    const uint32_t kMaxEventCount;
    EngineEvent*   fBuffer;
    CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineEventPort)
};
// -----------------------------------------------------------------------
/*!
 * Carla Engine client.\n
 * Each plugin requires one client from the engine (created via CarlaEngine::addPort()).\n
 * \note This is a virtual class, each engine type provides its own funtionality.
 */
class CarlaEngineClient
{
public:
    /*!
     * The constructor, protected.\n
     * All constructor parameters are constant and will never change in the lifetime of the client.\n
     * Client starts in deactivated state.
     */
    CarlaEngineClient(const CarlaBackend::EngineType engineType, const CarlaBackend::ProcessMode processMode);
    /*!
     * The destructor.
     */
    virtual ~CarlaEngineClient();
    /*!
     * Activate this client.\n
     * Client must be deactivated before calling this function.
     */
    virtual void activate();
    /*!
     * Deactivate this client.\n
     * Client must be activated before calling this function.
     */
    virtual void deactivate();
    /*!
     * Check if the client is activated.
     */
    virtual bool isActive() const;
    /*!
     * Check if the client is ok.\n
     * Plugins will refuse to instantiate if this returns false.
     * \note This is always true in rack and patchbay processing modes.
     */
    virtual bool isOk() const;
    /*!
     * Get the current latency, in samples.
     */
    virtual uint32_t getLatency() const;
    /*!
     * Change the client's latency.
     */
    virtual void setLatency(const uint32_t samples);
    /*!
     * Add a new port of type \a portType.
     * \note This function does nothing in rack processing mode since ports are static there (2 audio + 1 event for both input and output).
     */
    virtual const CarlaEnginePort* addPort(const EnginePortType portType, const char* const name, const bool isInput);
protected:
    const EngineType  kEngineType;
    const ProcessMode kProcessMode;
    bool     fActive;
    uint32_t fLatency;
private:
    CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngineClient)
};
// -----------------------------------------------------------------------
/*!
 * Protected data used in CarlaEngine.
 * Non-engine code MUST NEVER have direct access to this.
 */
struct CarlaEngineProtectedData;
/*!
 * Carla Engine.
 * \note This is a virtual class for all available engine types available in Carla.
 */
class CarlaEngine
{
protected:
    /*!
     * The constructor, protected.\n
     * \note This only initializes engine data, it doesn't initialize the engine itself.
     */
    CarlaEngine();
public:
    /*!
     * The decontructor.
     * The engine must have been closed before this happens.
     */
    virtual ~CarlaEngine();
    // -------------------------------------------------------------------
    // Static values and calls
    /*!
     * Get the number of available engine drivers.
     */
    static unsigned int getDriverCount();
    /*!
     * Get the name of the engine driver at \a index.
     */
    static const char* getDriverName(unsigned int index);
    /*!
     * Create a new engine, using driver \a driverName.\n
     * Returned variable must be deleted when no longer needed.
     */
    static CarlaEngine* newDriverByName(const char* const driverName);
    // -------------------------------------------------------------------
    // Constant values
    /*!
     * Maximum client name size.
     */
    virtual unsigned int maxClientNameSize() const;
    /*!
     * Maximum port name size.
     */
    virtual unsigned int maxPortNameSize() const;
    /*!
     * Current number of plugins loaded.
     */
    unsigned int currentPluginCount() const;
    /*!
     * Maximum number of loadable plugins allowed.
     * \note This function returns 0 if engine is not started.
     */
    unsigned int maxPluginNumber() const;
    // -------------------------------------------------------------------
    // Virtual, per-engine type calls
    /*!
     * Initialize engine, using \a clientName.
     */
    virtual bool init(const char* const clientName);
    /*!
     * Close engine.
     */
    virtual bool close();
    /*!
     * Idle engine.
     */
    virtual void idle();
    /*!
     * Check if engine is running.
     */
    virtual bool isRunning() const = 0;
    /*!
     * Check if engine is running offline (aka freewheel mode).
     */
    virtual bool isOffline() const = 0;
    /*!
     * Get engine type.
     */
    virtual EngineType type() const = 0;
    /*!
     * Add new engine client.
     * \note This must only be called within a plugin class.
     */
    virtual CarlaEngineClient* addClient(CarlaPlugin* const plugin);
    // -------------------------------------------------------------------
    // Plugin management
    /*!
     * Add new plugin.
     */
    bool addPlugin(const BinaryType btype, const PluginType ptype, const char* const filename, const char* const name, const char* const label, const void* const extra = nullptr);
    /*!
     * Add new plugin, using native binary type.
     */
    bool addPlugin(const PluginType ptype, const char* const filename, const char* const name, const char* const label, const void* const extra = nullptr)
    {
        return addPlugin(BINARY_NATIVE, ptype, filename, name, label, extra);
    }
    /*!
     * Remove plugin with id \a id.
     */
    bool removePlugin(const unsigned int id);
    /*!
     * Remove all plugins.
     */
    void removeAllPlugins();
    /*!
     * Get plugin with id \a id.
     */
    CarlaPlugin* getPlugin(const unsigned int id) const;
    /*!
     * Get plugin with id \a id, faster unchecked version.
     */
    CarlaPlugin* getPluginUnchecked(const unsigned int id) const;
    /*!
     * Get a unique plugin name within the engine.\n
     * Returned variable must NOT be free'd.
     */
    const char* getNewUniquePluginName(const char* const name);
    // -------------------------------------------------------------------
    // Project management
    /*!
     * Load \a filename session.
     * \note Already loaded plugins are not removed; call removeAllPlugins() first if needed.
     */
    bool loadProject(const char* const filename);
    /*!
     * Save current session to \a filename.
     */
    bool saveProject(const char* const filename);
    // -------------------------------------------------------------------
    // Information (base)
    /*!
     * Get current buffer size.
     */
    uint32_t getBufferSize() const
    {
        return fBufferSize;
    }
    /*!
     * Get current sample rate.
     */
    double getSampleRate() const
    {
        return fSampleRate;
    }
    /*!
     * Get engine name.
     */
    const char* getName() const
    {
        return (const char*)fName;
    }
    /*!
     * Get the engine options (read-only).
     */
    const EngineOptions& getOptions() const
    {
        return fOptions;
    }
    /*!
     * TODO.
     */
    ProcessMode getProccessMode() const
    {
        return fOptions.processMode;
    }
    /*!
     * Get current Time information (read-only).
     */
    const EngineTimeInfo& getTimeInfo() const
    {
        return fTimeInfo;
    }
    // -------------------------------------------------------------------
    // Information (peaks)
    /*!
     * TODO.
     */
    float getInputPeak(const unsigned int pluginId, const unsigned short id) const;
    /*!
     * TODO.
     */
    float getOutputPeak(const unsigned int pluginId, const unsigned short id) const;
    // -------------------------------------------------------------------
    // Callback
    /*!
     * TODO.
     */
    void callback(const CallbackType action, const unsigned int pluginId, const int value1, const int value2, const float value3, const char* const valueStr);
    /*!
     * TODO.
     */
    void setCallback(const CallbackFunc func, void* const ptr);
    // -------------------------------------------------------------------
    // Error handling
    /*!
     * Get last error.
     */
    const char* getLastError() const;
    /*!
     * Set last error.
     */
    void setLastError(const char* const error);
    // -------------------------------------------------------------------
    // Misc
    /*!
     * Tell the engine it's about to close.\n
     * This is used to prevent the engine thread(s) from reactivating.
     */
    void setAboutToClose();
    /*!
     * Safely block-wait until the current proccessing callback ends.
     */
    void waitForProccessEnd();
#ifndef BUILD_BRIDGE
    // -------------------------------------------------------------------
    // Options
    /*!
     * Get the engine options as process environment.
     */
    const QProcessEnvironment& getOptionsAsProcessEnvironment() const;
    /*!
     * Set the engine option \a option.
     */
    void setOption(const OptionsType option, const int value, const char* const valueStr);
#endif
    // -------------------------------------------------------------------
    // OSC Stuff
#ifdef BUILD_BRIDGE
    /*!
     * Check if OSC bridge is registered.
     */
    bool isOscBridgeRegistered() const;
#else
    /*!
     * Check if OSC controller is registered.
     */
    bool isOscControlRegistered() const;
#endif
    /*!
     * Idle OSC.
     */
    void idleOsc();
    /*!
     * Get OSC TCP server path.
     */
    const char* getOscServerPathTCP() const;
    /*!
     * Get OSC UDP server path.
     */
    const char* getOscServerPathUDP() const;
#ifdef BUILD_BRIDGE
    /*!
     * Set OSC bridge data.
     */
    void setOscBridgeData(const CarlaOscData* const oscData);
#endif
    // -------------------------------------
protected:
    static const unsigned short MAX_PEAKS = 2;
    uint32_t fBufferSize;
    double   fSampleRate;
    CarlaString fName;
    EngineOptions  fOptions;
    EngineTimeInfo fTimeInfo;
    CarlaEngineProtectedData* const kData;
    /*!
     * Report to all plugins about buffer size change.
     */
    void bufferSizeChanged(const uint32_t newBufferSize);
    /*!
     * Report to all plugins about sample rate change.\n
     * This is not supported on all plugin types, on which case they will be re-initiated.\n
     * TODO: Not implemented yet.
     */
    void sampleRateChanged(const double newSampleRate);
    /*!
     * TODO.
     */
    void proccessPendingEvents();
    /*!
     * TODO.
     */
    void setPeaks(const unsigned int pluginId, float inPeaks[MAX_PEAKS], float outPeaks[MAX_PEAKS]);
#ifndef BUILD_BRIDGE
    // Rack mode data
    EngineEvent* getRackEventBuffer(const bool isInput);
    /*!
     * Proccess audio buffer in rack mode.
     */
    void processRack(float* inBuf[2], float* outBuf[2], const uint32_t frames);
    /*!
     * Proccess audio buffer in patchbay mode.
     * In \a bufCount, [0]=inBufCount and [1]=outBufCount
     */
    void processPatchbay(float** inBuf, float** outBuf, const uint32_t bufCount[2], const uint32_t frames);
#endif
private:
#ifdef WANT_JACK
    static CarlaEngine* newJack();
#endif
#ifdef WANT_RTAUDIO
    enum RtAudioApi {
        RTAUDIO_DUMMY        = 0,
        RTAUDIO_LINUX_ALSA   = 1,
        RTAUDIO_LINUX_PULSE  = 2,
        RTAUDIO_LINUX_OSS    = 3,
        RTAUDIO_UNIX_JACK    = 4,
        RTAUDIO_MACOSX_CORE  = 5,
        RTAUDIO_WINDOWS_ASIO = 6,
        RTAUDIO_WINDOWS_DS   = 7
    };
    static CarlaEngine* newRtAudio(const RtAudioApi api);
    static unsigned int getRtAudioApiCount();
    static const char*  getRtAudioApiName(unsigned int index);
#endif
public:
#ifdef BUILD_BRIDGE
    void osc_send_peaks(CarlaPlugin* const plugin);
    void osc_send_bridge_audio_count(const int32_t ins, const int32_t outs, const int32_t total);
    void osc_send_bridge_midi_count(const int32_t ins, const int32_t outs, const int32_t total);
    void osc_send_bridge_parameter_count(const int32_t ins, const int32_t outs, const int32_t total);
    void osc_send_bridge_program_count(const int32_t count);
    void osc_send_bridge_midi_program_count(const int32_t count);
    void osc_send_bridge_plugin_info(const int32_t category, const int32_t hints, const char* const name, const char* const label, const char* const maker, const char* const copyright, const int64_t uniqueId);
    void osc_send_bridge_parameter_info(const int32_t index, const char* const name, const char* const unit);
    void osc_send_bridge_parameter_data(const int32_t index, const int32_t type, const int32_t rindex, const int32_t hints, const int32_t midiChannel, const int32_t midiCC);
    void osc_send_bridge_parameter_ranges(const int32_t index, const double def, const double min, const double max, const double step, const double stepSmall, const double stepLarge);
    void osc_send_bridge_program_info(const int32_t index, const char* const name);
    void osc_send_bridge_midi_program_info(const int32_t index, const int32_t bank, const int32_t program, const char* const label);
    void osc_send_bridge_configure(const char* const key, const char* const value);
    void osc_send_bridge_set_parameter_value(const int32_t index, const double value);
    void osc_send_bridge_set_default_value(const int32_t index, const double value);
    void osc_send_bridge_set_program(const int32_t index);
    void osc_send_bridge_set_midi_program(const int32_t index);
    void osc_send_bridge_set_custom_data(const char* const type, const char* const key, const char* const value);
    void osc_send_bridge_set_chunk_data(const char* const chunkFile);
    void osc_send_bridge_set_inpeak(const int32_t portId);
    void osc_send_bridge_set_outpeak(const int32_t portId);
#else
    void osc_send_peaks(CarlaPlugin* const plugin, const unsigned short id);
    void osc_send_control_add_plugin_start(const int32_t pluginId, const char* const pluginName);
    void osc_send_control_add_plugin_end(const int32_t pluginId);
    void osc_send_control_remove_plugin(const int32_t pluginId);
    void osc_send_control_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_control_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_control_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_control_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_control_set_parameter_midi_cc(const int32_t pluginId, const int32_t index, const int32_t cc);
    void osc_send_control_set_parameter_midi_channel(const int32_t pluginId, const int32_t index, const int32_t channel);
    void osc_send_control_set_parameter_value(const int32_t pluginId, const int32_t index, const double value);
    void osc_send_control_set_default_value(const int32_t pluginId, const int32_t index, const double value);
    void osc_send_control_set_program(const int32_t pluginId, const int32_t index);
    void osc_send_control_set_program_count(const int32_t pluginId, const int32_t count);
    void osc_send_control_set_program_name(const int32_t pluginId, const int32_t index, const char* const name);
    void osc_send_control_set_midi_program(const int32_t pluginId, const int32_t index);
    void osc_send_control_set_midi_program_count(const int32_t pluginId, const int32_t count);
    void osc_send_control_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_control_note_on(const int32_t pluginId, const int32_t channel, const int32_t note, const int32_t velo);
    void osc_send_control_note_off(const int32_t pluginId, const int32_t channel, const int32_t note);
    void osc_send_control_set_input_peak_value(const int32_t pluginId, const int32_t portId);
    void osc_send_control_set_output_peak_value(const int32_t pluginId, const int32_t portId);
    void osc_send_control_exit();
#endif
private:
    friend class CarlaEngineEventPort;
    CARLA_DECLARE_NON_COPYABLE_WITH_LEAK_DETECTOR(CarlaEngine)
};
// -----------------------------------------------------------------------
/**@}*/
CARLA_BACKEND_END_NAMESPACE
#endif // __CARLA_ENGINE_HPP__
 |