/* * Copyright (c) 2014-2020 Christian Schoenebeck * * http://www.linuxsampler.org * * This file is part of LinuxSampler and released under the same terms. * See README file for details. */ #ifndef LS_INSTRUMENT_SCRIPT_VM_H #define LS_INSTRUMENT_SCRIPT_VM_H #include "../../common/global.h" #include "../../common/ConstCapacityArray.h" #include "../../scriptvm/ScriptVM.h" #include "Event.h" #include "../../common/Pool.h" #include "InstrumentScriptVMFunctions.h" #include "InstrumentScriptVMDynVars.h" /** * Amount of bits on the left hand side of all pool_element_id_t numbers (i.e. * event_id_t, note_id_t) being reserved for script VM implementation internal * purposes. * * Right now there is only one bit reserved, which allows the VM (and its * built-in functions) to distinguish user supplied @c Event IDs (event_id_t) * from @c Note IDs (note_id_t). */ #define INSTR_SCRIPT_EVENT_ID_RESERVED_BITS 1 /** * Used by InstrScriptIDType_T to initialize its constants at compile time. * * This macro is already ready to initialize additional members for * InstrScriptIDType_T (that is more than the currently two enum constants). * Just keep in mind that you also have to increase * INSTR_SCRIPT_EVENT_ID_RESERVED_BITS when you add more! * * @param x - sequential consecutive number (starting with zero) */ #define INSTR_SCRIPT_ID_TYPE_FLAG(x) \ (x << (sizeof(pool_element_id_t) * 8 - INSTR_SCRIPT_EVENT_ID_RESERVED_BITS)) /** * These flags are used to distinguish the individual ID Types in script scope * from each other. They are added as most left bit(s) to each ID in script * scope. */ enum InstrScriptIDType_T { /** * Used to mark IDs (in script scope) to actually be a MIDI event ID. */ INSTR_SCRIPT_EVENT_ID_FLAG = INSTR_SCRIPT_ID_TYPE_FLAG(0LL), /** * Used to mark IDs (in script scope) to actually be a note ID. */ INSTR_SCRIPT_NOTE_ID_FLAG = INSTR_SCRIPT_ID_TYPE_FLAG(1LL), }; #define INSTR_SCRIPT_EVENT_GROUPS 28 #define EVENT_STATUS_INACTIVE 0 #define EVENT_STATUS_NOTE_QUEUE 1 enum { EVENT_PAR_NOTE = 1, EVENT_PAR_VELOCITY, EVENT_PAR_VOLUME, EVENT_PAR_TUNE, EVENT_PAR_0 = 1024, EVENT_PAR_1, EVENT_PAR_2, EVENT_PAR_3, }; enum { CALLBACK_STATUS_TERMINATED = 0, CALLBACK_STATUS_QUEUE = 1, CALLBACK_STATUS_RUNNING = (1 << 1) }; namespace LinuxSampler { class AbstractEngineChannel; struct InstrumentScript; /** @brief Convert IDs between script scope and engine internal scope. * * This class is used to translate unique IDs of events between script * scope and sampler engine internal scope, that is: * @code * int (script scope) -> event_id_t (engine internal scope) * int (script scope) -> note_id_t (engine internal scope) * @endcode * and vice versa: * @code * event_id_t (engine internal scope) -> int (script scope) * note_id_t (engine internal scope) -> int (script scope) * @endcode * This is required because engine internally i.e. notes and regular events * are using their own, separate ID generating pool, and their ID number * set may thus overlap and historically there were built-in script * functions in KSP which allow to pass both regular MIDI event IDs, as well * as Note IDs. So we must be able to distinguish between them in our * built-in script function implementations. * * @see INSTR_SCRIPT_EVENT_ID_RESERVED_BITS */ class ScriptID { public: enum type_t { EVENT, ///< ID is actually an event ID NOTE, ///< ID is actually a note ID }; /** * Construct a ScriptID object with an ID from script scope. */ ScriptID(vmint id) : m_id(id) {} /** * Returns a ScriptID object constructed with an event ID from engine * internal scope. */ inline static ScriptID fromEventID(event_id_t id) { return ScriptID(INSTR_SCRIPT_EVENT_ID_FLAG | id); } /** * Returns a ScriptID object constructed with a note ID from engine * internal scope. */ inline static ScriptID fromNoteID(note_id_t id) { return ScriptID(INSTR_SCRIPT_NOTE_ID_FLAG | id); } /** * Whether the ID reflected by this ScriptID object is actually a note * ID or rather an event ID. */ inline type_t type() const { return (m_id & INSTR_SCRIPT_NOTE_ID_FLAG) ? NOTE : EVENT; } inline bool isNoteID() const { return type() == NOTE; } inline bool isEventID() const { return type() == EVENT; } /** * Returns event ID (for engine internal scope) of the ID reflected by * this ScriptID object, it returns 0 (being an invalid ID) if the ID * reflected by this ScriptID object is not an event ID. */ inline event_id_t eventID() const { switch (type()) { case EVENT: return m_id; default: return 0; // not an event id, return invalid ID } } /** * Returns note ID (for engine internal scope) of the ID reflected by * this ScriptID object, it returns 0 (being an invalid ID) if the ID * reflected by this ScriptID object is not a note ID. */ inline note_id_t noteID() const { switch (type()) { case NOTE: return ~INSTR_SCRIPT_NOTE_ID_FLAG & m_id; default: return 0; // not a note id, return invalid ID } } /** * Integer cast operator, which returns an ID number of this ScripID * object intended for script scope. */ inline operator vmint() const { return m_id; } private: vmint m_id; }; /** @brief List of Event IDs. * * Used for built-in script functions: * by_marks(), set_event_mark(), delete_event_mark() */ class EventGroup : protected ConstCapacityArray { public: EventGroup() : ConstCapacityArray(CONFIG_MAX_EVENTS_PER_FRAGMENT), m_script(NULL) {} void insert(vmint eventID); void erase(vmint eventID); void setScript(InstrumentScript* pScript) { m_script = pScript; } inline size_t size() const { return ConstCapacityArray::size(); } inline void clear() { ConstCapacityArray::clear(); } inline vmint& operator[](ssize_t index) { return ConstCapacityArray::operator[](index); } inline const vmint& operator[](ssize_t index) const { return ConstCapacityArray::operator[](index); } protected: InstrumentScript* m_script; }; /** @brief Real-time instrument script VM representation. * * Holds the VM representation of all event handlers of the currently loaded * script, ready to be executed by the sampler engine. * * Even thought scripts (or to be more specific their event handler objects) * are shared between sampler engine channels, the InstrumentScript struct * instances though are not shared. Each sampler engine channel has its own * instance of a InstrumentScript struct. That's important, because this * struct also holds engine channel local informations, for example the * events that occured on the respective engine channel. */ struct InstrumentScript { VMParserContext* parserContext; ///< VM represenation of the currently loaded script or NULL if not script was loaded. Note that it is also not NULL if parser errors occurred! bool bHasValidScript; ///< True in case there is a valid script currently loaded, false if script processing shall be skipped. VMEventHandler* handlerInit; ///< VM representation of script's initilization callback or NULL if current script did not define such an init handler. VMEventHandler* handlerNote; ///< VM representation of script's MIDI note on callback or NULL if current script did not define such an event handler. VMEventHandler* handlerRelease; ///< VM representation of script's MIDI note off callback or NULL if current script did not define such an event handler. VMEventHandler* handlerController; ///< VM representation of script's MIDI controller callback or NULL if current script did not define such an event handler. VMEventHandler* handlerRpn; ///< VM representation of script's MIDI RPN event callback or NULL if current script did not define such an event handler. VMEventHandler* handlerNrpn; ///< VM representation of script's MIDI NRPN event callback or NULL if current script did not define such an event handler. Pool* pEvents; ///< Pool of all available script execution instances. ScriptEvents available to be allocated from the Pool are currently unused / not executiong, whereas the ScriptEvents allocated on the list are currently suspended / have not finished execution yet (@see pKeyEvents). RTList* pKeyEvents[128]; ///< Stores previously finished executed "note on" script events for the respective active note/key as long as the key/note is active. This is however only done if there is a "note" script event handler and a "release" script event handler defined in the script and both handlers use (reference) polyphonic variables. If that is not the case, then this list is not used at all. So the purpose of pKeyEvents is only to implement preserving/passing polyphonic variable data from "on note .. end on" script block to the respective "on release .. end on" script block. RTAVLTree suspendedEvents; ///< Contains pointers to all suspended events, sorted by time when those script events are to be resumed next. AbstractEngineChannel* pEngineChannel; String code; ///< Source code of the instrument script. Used in case the sampler engine is changed, in that case a new ScriptVM object is created for the engine and VMParserContext object for this script needs to be recreated as well. Thus the script is then parsed again by passing the source code to recreate the parser context. EventGroup eventGroups[INSTR_SCRIPT_EVENT_GROUPS]; ///< Used for built-in script functions: by_event_marks(), set_event_mark(), delete_event_mark(). InstrumentScript(AbstractEngineChannel* pEngineChannel); ~InstrumentScript(); void load(const String& text); void unload(); void resetAll(); void resetEvents(); }; /** @brief Real-time instrument script virtual machine. * * Extends the core ScriptVM implementation with MIDI specific built-in * script functions and MIDI specific built-in script variables required * for MIDI processing by instrument script for all sampler engine * implementations (sampler formats) of this sampler. * * Note that this class is currently re-entrant safe, but @b not thread * safe! See also comments of base class ScriptVM regarding this issue. */ class InstrumentScriptVM : public ScriptVM { public: InstrumentScriptVM(); VMExecStatus_t exec(VMParserContext* parserCtx, ScriptEvent* event); VMFunction* functionByName(const String& name) OVERRIDE; std::map builtInIntVariables() OVERRIDE; std::map builtInIntArrayVariables() OVERRIDE; std::map builtInConstIntVariables() OVERRIDE; std::map builtInDynamicVariables() OVERRIDE; protected: ScriptEvent* m_event; ///< The event currently executed by exec(). // built-in script variables VMInt8Array m_CC; VMInt8RelPtr m_CC_NUM; VMIntRelPtr m_EVENT_ID; VMInt8RelPtr m_EVENT_NOTE; VMInt8RelPtr m_EVENT_VELOCITY; VMInt8Array m_KEY_DOWN; VMInt16RelPtr m_RPN_ADDRESS; // used for both RPN and NRPN events VMInt16RelPtr m_RPN_VALUE; // used for both RPN and NRPN events //VMIntArray m_POLY_AT; //TODO: ... //int m_POLY_AT_NUM; //TODO: ... VMInt32RelPtr m_NI_CALLBACK_TYPE; VMInt8RelPtr m_NKSP_IGNORE_WAIT; VMIntRelPtr m_NKSP_CALLBACK_PARENT_ID; // built-in script functions InstrumentScriptVMFunction_play_note m_fnPlayNote; InstrumentScriptVMFunction_set_controller m_fnSetController; InstrumentScriptVMFunction_ignore_event m_fnIgnoreEvent; InstrumentScriptVMFunction_ignore_controller m_fnIgnoreController; InstrumentScriptVMFunction_note_off m_fnNoteOff; InstrumentScriptVMFunction_set_event_mark m_fnSetEventMark; InstrumentScriptVMFunction_delete_event_mark m_fnDeleteEventMark; InstrumentScriptVMFunction_by_marks m_fnByMarks; InstrumentScriptVMFunction_change_vol m_fnChangeVol; InstrumentScriptVMFunction_change_vol_time m_fnChangeVolTime; InstrumentScriptVMFunction_change_tune m_fnChangeTune; InstrumentScriptVMFunction_change_tune_time m_fnChangeTuneTime; InstrumentScriptVMFunction_change_pan m_fnChangePan; InstrumentScriptVMFunction_change_pan_time m_fnChangePanTime; InstrumentScriptVMFunction_change_pan_curve m_fnChangePanCurve; InstrumentScriptVMFunction_change_cutoff m_fnChangeCutoff; InstrumentScriptVMFunction_change_reso m_fnChangeReso; InstrumentScriptVMFunction_change_attack m_fnChangeAttack; InstrumentScriptVMFunction_change_decay m_fnChangeDecay; InstrumentScriptVMFunction_change_sustain m_fnChangeSustain; InstrumentScriptVMFunction_change_release m_fnChangeRelease; InstrumentScriptVMFunction_change_cutoff_attack m_fnChangeCutoffAttack; InstrumentScriptVMFunction_change_cutoff_decay m_fnChangeCutoffDecay; InstrumentScriptVMFunction_change_cutoff_sustain m_fnChangeCutoffSustain; InstrumentScriptVMFunction_change_cutoff_release m_fnChangeCutoffRelease; InstrumentScriptVMFunction_change_amp_lfo_depth m_fnChangeAmpLFODepth; InstrumentScriptVMFunction_change_amp_lfo_freq m_fnChangeAmpLFOFreq; InstrumentScriptVMFunction_change_cutoff_lfo_depth m_fnChangeCutoffLFODepth; InstrumentScriptVMFunction_change_cutoff_lfo_freq m_fnChangeCutoffLFOFreq; InstrumentScriptVMFunction_change_pitch_lfo_depth m_fnChangePitchLFODepth; InstrumentScriptVMFunction_change_pitch_lfo_freq m_fnChangePitchLFOFreq; InstrumentScriptVMFunction_change_note m_fnChangeNote; InstrumentScriptVMFunction_change_velo m_fnChangeVelo; InstrumentScriptVMFunction_fork m_fnFork; InstrumentScriptVMFunction_event_status m_fnEventStatus; InstrumentScriptVMFunction_wait m_fnWait2; InstrumentScriptVMFunction_stop_wait m_fnStopWait; InstrumentScriptVMFunction_abort m_fnAbort; InstrumentScriptVMFunction_fade_in m_fnFadeIn; InstrumentScriptVMFunction_fade_out m_fnFadeOut; InstrumentScriptVMFunction_change_vol_curve m_fnChangeVolCurve; InstrumentScriptVMFunction_change_tune_curve m_fnChangeTuneCurve; InstrumentScriptVMFunction_get_event_par m_fnGetEventPar; InstrumentScriptVMFunction_set_event_par m_fnSetEventPar; InstrumentScriptVMFunction_change_play_pos m_fnChangePlayPos; InstrumentScriptVMFunction_callback_status m_fnCallbackStatus; InstrumentScriptVMDynVar_ENGINE_UPTIME m_varEngineUptime; InstrumentScriptVMDynVar_NI_CALLBACK_ID m_varCallbackID; InstrumentScriptVMDynVar_ALL_EVENTS m_varAllEvents; InstrumentScriptVMDynVar_NKSP_CALLBACK_CHILD_ID m_varCallbackChildID; friend class InstrumentScriptVMFunction_play_note; friend class InstrumentScriptVMFunction_set_controller; friend class InstrumentScriptVMFunction_ignore_event; friend class InstrumentScriptVMFunction_ignore_controller; friend class InstrumentScriptVMFunction_note_off; friend class InstrumentScriptVMFunction_set_event_mark; friend class InstrumentScriptVMFunction_delete_event_mark; friend class InstrumentScriptVMFunction_by_marks; friend class InstrumentScriptVMFunction_change_vol; friend class InstrumentScriptVMFunction_change_vol_time; friend class InstrumentScriptVMFunction_change_tune; friend class InstrumentScriptVMFunction_change_tune_time; friend class InstrumentScriptVMFunction_change_pan; friend class InstrumentScriptVMFunction_change_pan_time; friend class InstrumentScriptVMFunction_change_pan_curve; friend class InstrumentScriptVMFunction_change_cutoff; friend class InstrumentScriptVMFunction_change_reso; friend class InstrumentScriptVMFunction_change_attack; friend class InstrumentScriptVMFunction_change_decay; friend class InstrumentScriptVMFunction_change_release; friend class InstrumentScriptVMFunction_change_sustain; friend class InstrumentScriptVMFunction_change_cutoff_attack; friend class InstrumentScriptVMFunction_change_cutoff_decay; friend class InstrumentScriptVMFunction_change_cutoff_release; friend class InstrumentScriptVMFunction_change_cutoff_sustain; friend class VMChangeSynthParamFunction; friend class InstrumentScriptVMFunction_change_amp_lfo_depth; friend class InstrumentScriptVMFunction_change_amp_lfo_freq; friend class InstrumentScriptVMFunction_change_cutoff_lfo_depth; friend class InstrumentScriptVMFunction_change_cutoff_lfo_freq; friend class InstrumentScriptVMFunction_change_pitch_lfo_depth; friend class InstrumentScriptVMFunction_change_pitch_lfo_freq; friend class VMChangeFadeCurveFunction; friend class InstrumentScriptVMFunction_fade_in; friend class InstrumentScriptVMFunction_fade_out; friend class InstrumentScriptVMFunction_get_event_par; friend class InstrumentScriptVMFunction_set_event_par; friend class InstrumentScriptVMFunction_change_note; friend class InstrumentScriptVMFunction_change_velo; friend class InstrumentScriptVMFunction_change_play_pos; friend class InstrumentScriptVMFunction_event_status; friend class InstrumentScriptVMFunction_wait; friend class InstrumentScriptVMFunction_stop_wait; friend class InstrumentScriptVMFunction_abort; friend class InstrumentScriptVMFunction_fork; friend class InstrumentScriptVMFunction_callback_status; friend class InstrumentScriptVMDynVar_ENGINE_UPTIME; friend class InstrumentScriptVMDynVar_NI_CALLBACK_ID; friend class InstrumentScriptVMDynVar_ALL_EVENTS; friend class InstrumentScriptVMDynVar_NKSP_CALLBACK_CHILD_ID; }; } // namespace LinuxSampler #endif // LS_INSTRUMENT_SCRIPT_VM_H