// Copyright 2007-2024 David Robillard // SPDX-License-Identifier: ISC #ifndef JALV_COMM_H #define JALV_COMM_H #include "attributes.h" #include "types.h" #include "lv2/atom/atom.h" #include "lv2/urid/urid.h" #include "zix/ring.h" #include "zix/status.h" #include // Communication between the audio and main threads via rings JALV_BEGIN_DECLS /// Type of an internal message in a communication ring typedef enum { NO_MESSAGE, ///< Sentinel type for uninitialized messages CONTROL_PORT_CHANGE, ///< Value change for a control port (float) EVENT_TRANSFER, ///< Event transfer for a sequence port (atom) LATENCY_CHANGE, ///< Change to plugin latency STATE_REQUEST, ///< Request for a plugin state update (no payload) RUN_STATE_CHANGE, ///< Request to pause or resume running } JalvMessageType; /** Message between the audio thread and the main thread. This is the general header for any type of message in a communication ring. The type determines how the message must be handled by the receiver. This header is followed immediately by `size` bytes of data in the ring. */ typedef struct { JalvMessageType type; ///< Type of this message uint32_t size; ///< Size of payload following this header in bytes } JalvMessageHeader; /** The payload of a CONTROL_PORT_CHANGE message. This message has a fixed sized, and is described in its entirety by this struct. */ typedef struct { uint32_t port_index; ///< Control port index float value; ///< Control value } JalvControlChange; /** The start of the payload of an EVENT_TRANSFER message. This message has a variable size, the start described by this struct is followed immediately by `atom.size` bytes of data (the atom body). */ typedef struct { uint32_t port_index; ///< Sequence port index LV2_Atom atom; ///< Event payload header } JalvEventTransfer; /** The payload of a LATENCY_CHANGE message. This message has a fixed sized, and is described in its entirety by this struct. */ typedef struct { float value; ///< Latency in frames at the current sample rate } JalvLatencyChange; /** The payload of a RUN_STATE_CHANGE message. This message has a fixed sized, and is described in its entirety by this struct. */ typedef struct { JalvRunState state; ///< Run state to change to } JalvRunStateChange; /** Write a message in two parts to a ring. This is used to conveniently write a message with a fixed-size header and possibly variably-sized body in a single call. @param target Communication ring (jalv->plugin_to_ui or jalv->ui_to_plugin). @param header Pointer to start of header data. @param header_size Size of header in bytes. @param body Pointer to start of body data. @param body_size Size of body in bytes. @return 0 on success, non-zero on failure (overflow). */ ZixStatus jalv_write_split_message(ZixRing* target, const void* header, uint32_t header_size, const void* body, uint32_t body_size); /** Write a port event using the atom:eventTransfer protocol. This is used to transfer atoms between the plugin and UI via sequence ports. @param target Communication ring (jalv->plugin_to_ui or jalv->ui_to_plugin). @param port_index Index of the port this change is for. @param size Size of body in bytes. @param type Atom type URID. @param body Atom body. @return 0 on success, non-zero on failure (overflow). */ ZixStatus jalv_write_event(ZixRing* target, uint32_t port_index, uint32_t size, LV2_URID type, const void* body); /** Write a control port change using the default (0) protocol. This is used to transfer control port value changes between the plugin and UI. @param target Communication ring (jalv->plugin_to_ui or jalv->ui_to_plugin). @param port_index Index of the port this change is for. @param value New control port value. @return 0 on success, non-zero on failure (overflow). */ ZixStatus jalv_write_control(ZixRing* target, uint32_t port_index, float value); JALV_END_DECLS #endif // JALV_COMM_H