More generic RT-safe implementation of LV2 properties.
[ardour.git] / libs / ardour / ardour / lv2_plugin.h
1 /*
2     Copyright (C) 2008-2012 Paul Davis
3     Author: David Robillard
4
5     This program is free software; you can redistribute it and/or modify
6     it under the terms of the GNU General Public License as published by
7     the Free Software Foundation; either version 2 of the License, or
8     (at your option) any later version.
9
10     This program is distributed in the hope that it will be useful,
11     but WITHOUT ANY WARRANTY; without even the implied warranty of
12     MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
13     GNU General Public License for more details.
14
15     You should have received a copy of the GNU General Public License
16     along with this program; if not, write to the Free Software
17     Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
18 */
19
20 #ifndef __ardour_lv2_plugin_h__
21 #define __ardour_lv2_plugin_h__
22
23 #include <set>
24 #include <string>
25 #include <vector>
26 #include <boost/enable_shared_from_this.hpp>
27
28 #include "ardour/plugin.h"
29 #include "ardour/uri_map.h"
30 #include "ardour/worker.h"
31 #include "pbd/ringbuffer.h"
32
33 #ifndef PATH_MAX
34 #define PATH_MAX 1024
35 #endif
36
37 typedef struct LV2_Evbuf_Impl LV2_Evbuf;
38
39 namespace ARDOUR {
40
41 // a callback function for lilv_state_new_from_instance(). friend of LV2Plugin
42 // so we can pass an LV2Plugin* in user_data and access its private members.
43 const void* lv2plugin_get_port_value(const char* port_symbol,
44                                      void*       user_data,
45                                      uint32_t*   size,
46                                      uint32_t*   type);
47
48 class AudioEngine;
49 class Session;
50
51 class LIBARDOUR_API LV2Plugin : public ARDOUR::Plugin, public ARDOUR::Workee
52 {
53   public:
54         LV2Plugin (ARDOUR::AudioEngine& engine,
55                    ARDOUR::Session&     session,
56                    const void*          c_plugin,
57                    framecnt_t           sample_rate);
58         LV2Plugin (const LV2Plugin &);
59         ~LV2Plugin ();
60
61         std::string unique_id () const;
62         const char* uri () const;
63         const char* label () const;
64         const char* name () const;
65         const char* maker () const;
66
67         uint32_t    num_ports () const;
68         uint32_t    parameter_count () const;
69         float       default_value (uint32_t port);
70         framecnt_t  signal_latency () const;
71         void        set_parameter (uint32_t port, float val);
72         float       get_parameter (uint32_t port) const;
73         std::string get_docs() const;
74         std::string get_parameter_docs(uint32_t which) const;
75         int         get_parameter_descriptor (uint32_t which, ParameterDescriptor&) const;
76         uint32_t    nth_parameter (uint32_t port, bool& ok) const;
77
78         const void* extension_data (const char* uri) const;
79
80         const void* c_plugin();
81         const void* c_ui();
82         const void* c_ui_type();
83
84         bool is_external_ui () const;
85         bool is_external_kx () const;
86         bool ui_is_resizable () const;
87
88         const char* port_symbol (uint32_t port) const;
89         uint32_t    port_index (const char* symbol) const;
90
91         const LV2_Feature* const* features () { return _features; }
92
93         std::set<Evoral::Parameter> automatable () const;
94
95         void activate ();
96         void deactivate ();
97         void cleanup ();
98
99         int set_block_size (pframes_t /*nframes*/) { return 0; }
100
101         int connect_and_run (BufferSet& bufs,
102                              ChanMapping in, ChanMapping out,
103                              pframes_t nframes, framecnt_t offset);
104
105         std::string describe_parameter (Evoral::Parameter);
106         std::string state_node_name () const { return "lv2"; }
107
108         void print_parameter (uint32_t param,
109                               char*    buf,
110                               uint32_t len) const;
111
112         bool parameter_is_audio (uint32_t) const;
113         bool parameter_is_control (uint32_t) const;
114         bool parameter_is_event (uint32_t) const;
115         bool parameter_is_input (uint32_t) const;
116         bool parameter_is_output (uint32_t) const;
117         bool parameter_is_toggled (uint32_t) const;
118
119         boost::shared_ptr<Plugin::ScalePoints>
120         get_scale_points(uint32_t port_index) const;
121
122         void set_insert_info(const PluginInsert* insert);
123
124         int      set_state (const XMLNode& node, int version);
125         bool     save_preset (std::string uri);
126         void     remove_preset (std::string uri);
127         bool     load_preset (PresetRecord);
128         std::string current_preset () const;
129
130         bool has_editor () const;
131         bool has_message_output () const;
132
133         bool write_from_ui(uint32_t       index,
134                            uint32_t       protocol,
135                            uint32_t       size,
136                            const uint8_t* body);
137
138         typedef void UIMessageSink(void*       controller,
139                                    uint32_t    index,
140                                    uint32_t    size,
141                                    uint32_t    format,
142                                    const void* buffer);
143
144         void enable_ui_emission();
145         void emit_to_ui(void* controller, UIMessageSink sink);
146
147         Worker* worker() { return _worker; }
148
149         int work(uint32_t size, const void* data);
150         int work_response(uint32_t size, const void* data);
151
152         void set_property(uint32_t key, const Variant& value);
153         void get_supported_properties(std::vector<ParameterDescriptor>& descs);
154         void announce_property_values();
155
156         static URIMap _uri_map;
157
158         struct URIDs {
159                 uint32_t atom_Chunk;
160                 uint32_t atom_Path;
161                 uint32_t atom_Sequence;
162                 uint32_t atom_eventTransfer;
163                 uint32_t atom_URID;
164                 uint32_t atom_Blank;
165                 uint32_t atom_Object;
166                 uint32_t log_Error;
167                 uint32_t log_Note;
168                 uint32_t log_Warning;
169                 uint32_t midi_MidiEvent;
170                 uint32_t time_Position;
171                 uint32_t time_bar;
172                 uint32_t time_barBeat;
173                 uint32_t time_beatUnit;
174                 uint32_t time_beatsPerBar;
175                 uint32_t time_beatsPerMinute;
176                 uint32_t time_frame;
177                 uint32_t time_speed;
178                 uint32_t patch_Get;
179                 uint32_t patch_Set;
180                 uint32_t patch_property;
181                 uint32_t patch_value;
182         };
183
184         static URIDs urids;
185
186   private:
187         struct Impl;
188         Impl*         _impl;
189         void*         _module;
190         LV2_Feature** _features;
191         Worker*       _worker;
192         framecnt_t    _sample_rate;
193         float*        _control_data;
194         float*        _shadow_data;
195         float*        _defaults;
196         LV2_Evbuf**   _ev_buffers;
197         LV2_Evbuf**   _atom_ev_buffers;
198         float*        _bpm_control_port;  ///< Special input set by ardour
199         float*        _freewheel_control_port;  ///< Special input set by ardour
200         float*        _latency_control_port;  ///< Special output set by ardour
201         framepos_t    _next_cycle_start;  ///< Expected start frame of next run cycle
202         double        _next_cycle_speed;  ///< Expected start frame of next run cycle
203         PBD::ID       _insert_id;
204         uint32_t      _patch_port_in_index;
205         uint32_t      _patch_port_out_index;
206
207         friend const void* lv2plugin_get_port_value(const char* port_symbol,
208                                                     void*       user_data,
209                                                     uint32_t*   size,
210                                                     uint32_t*   type);
211
212         typedef enum {
213                 PORT_INPUT    = 1,       ///< Input port
214                 PORT_OUTPUT   = 1 << 1,  ///< Output port
215                 PORT_AUDIO    = 1 << 2,  ///< Audio (buffer of float)
216                 PORT_CONTROL  = 1 << 3,  ///< Control (single float)
217                 PORT_EVENT    = 1 << 4,  ///< Old event API event port
218                 PORT_SEQUENCE = 1 << 5,  ///< New atom API event port
219                 PORT_MIDI     = 1 << 6,  ///< Event port understands MIDI
220                 PORT_POSITION = 1 << 7,  ///< Event port understands position
221                 PORT_PATCHMSG = 1 << 8   ///< Event port supports patch:Message
222         } PortFlag;
223
224         typedef unsigned PortFlags;
225
226         std::vector<PortFlags>         _port_flags;
227         std::vector<size_t>            _port_minimumSize;
228         std::map<std::string,uint32_t> _port_indices;
229
230         /// Message send to/from UI via ports
231         struct UIMessage {
232                 uint32_t index;
233                 uint32_t protocol;
234                 uint32_t size;
235         };
236
237         bool write_to_ui(uint32_t       index,
238                          uint32_t       protocol,
239                          uint32_t       size,
240                          const uint8_t* body);
241
242         bool write_to(RingBuffer<uint8_t>* dest,
243                       uint32_t             index,
244                       uint32_t             protocol,
245                       uint32_t             size,
246                       const uint8_t*       body);
247
248         // Created on demand so the space is only consumed if necessary
249         RingBuffer<uint8_t>* _to_ui;
250         RingBuffer<uint8_t>* _from_ui;
251
252         typedef struct {
253                 const void* (*extension_data) (const char* uri);
254         } LV2_DataAccess;
255
256         LV2_DataAccess _data_access_extension_data;
257         LV2_Feature    _data_access_feature;
258         LV2_Feature    _instance_access_feature;
259         LV2_Feature    _make_path_feature;
260         LV2_Feature    _log_feature;
261         LV2_Feature    _work_schedule_feature;
262         LV2_Feature    _options_feature;
263         LV2_Feature    _def_state_feature;
264
265         // Options passed to plugin
266         int32_t _block_length;
267         int32_t _seq_size;
268
269         mutable unsigned _state_version;
270
271         bool _was_activated;
272         bool _has_state_interface;
273
274         const std::string plugin_dir () const;
275         const std::string scratch_dir () const;
276         const std::string file_dir () const;
277         const std::string state_dir (unsigned num) const;
278
279         static char* lv2_state_make_path (void*       host_data,
280                                           const char* path);
281
282         void init (const void* c_plugin, framecnt_t rate);
283         void allocate_atom_event_buffers ();
284         void run (pframes_t nsamples);
285
286         void latency_compute_run ();
287         std::string do_save_preset (std::string);
288         void do_remove_preset (std::string);
289         void find_presets ();
290         void add_state (XMLNode *) const;
291 };
292
293
294 class LIBARDOUR_API LV2PluginInfo : public PluginInfo , public boost::enable_shared_from_this<ARDOUR::LV2PluginInfo> {
295 public:
296         LV2PluginInfo (const char* plugin_uri);
297         ~LV2PluginInfo ();
298
299         static PluginInfoList* discover ();
300
301         PluginPtr load (Session& session);
302
303         char * _plugin_uri;
304 };
305
306 typedef boost::shared_ptr<LV2PluginInfo> LV2PluginInfoPtr;
307
308 } // namespace ARDOUR
309
310 #endif /* __ardour_lv2_plugin_h__ */