/*
- Copyright (C) 2006 Paul Davis
- Written by Taybin Rutkin
+ Copyright (C) 2006 Paul Davis
+ Written by Taybin Rutkin
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
#define __ardour_audio_unit_h__
#include <stdint.h>
+#include <boost/shared_ptr.hpp>
#include <list>
#include <set>
#include <string>
#include <vector>
+#include <map>
+
+#include "ardour/plugin.h"
-#include <ardour/plugin.h>
+#include <AudioUnit/AudioUnit.h>
+#include <AudioUnit/AudioUnitProperties.h>
+#include "AUParamInfo.h"
#include <boost/shared_ptr.hpp>
class AudioEngine;
class Session;
-class AUPlugin : public ARDOUR::Plugin
+struct LIBARDOUR_API AUParameterDescriptor : public ParameterDescriptor {
+ // additional fields to make operations more efficient
+ AudioUnitParameterID id;
+ AudioUnitScope scope;
+ AudioUnitElement element;
+ bool automatable;
+ AudioUnitParameterUnit au_unit;
+};
+
+class LIBARDOUR_API AUPlugin : public ARDOUR::Plugin
{
public:
- AUPlugin (AudioEngine& engine, Session& session, CAComponent* comp);
+ AUPlugin (AudioEngine& engine, Session& session, boost::shared_ptr<CAComponent> comp);
+ AUPlugin (const AUPlugin& other);
virtual ~AUPlugin ();
-
- uint32_t unique_id () const;
+
+ std::string unique_id () const;
const char * label () const;
const char * name () const { return _info->name.c_str(); }
- const char * maker () const;
+ const char * maker () const { return _info->creator.c_str(); }
uint32_t parameter_count () const;
float default_value (uint32_t port);
- jack_nframes_t latency () const;
+ samplecnt_t signal_latency() const;
void set_parameter (uint32_t which, float val);
float get_parameter (uint32_t which) const;
-
+
+ PluginOutputConfiguration possible_output () const { return _output_configs; }
+
int get_parameter_descriptor (uint32_t which, ParameterDescriptor&) const;
uint32_t nth_parameter (uint32_t which, bool& ok) const;
void activate ();
void deactivate ();
- void set_block_size (jack_nframes_t nframes);
-
- int connect_and_run (BufferSet& bufs, uint32_t& in, uint32_t& out, jack_nframes_t nframes, jack_nframes_t offset);
- std::set<uint32_t> automatable() const;
- void store_state (ARDOUR::PluginState&);
- void restore_state (ARDOUR::PluginState&);
- string describe_parameter (uint32_t);
- string state_node_name () const { return "audiounit"; }
+ void flush ();
+ int set_block_size (pframes_t nframes);
+
+ int connect_and_run (BufferSet& bufs,
+ samplepos_t start, samplepos_t end, double speed,
+ ChanMapping in, ChanMapping out,
+ pframes_t nframes, samplecnt_t offset);
+ std::set<Evoral::Parameter> automatable() const;
+ std::string describe_parameter (Evoral::Parameter);
+ IOPortDescription describe_io_port (DataType dt, bool input, uint32_t id) const;
+ std::string state_node_name () const { return "audiounit"; }
void print_parameter (uint32_t, char*, uint32_t len) const;
-
+
bool parameter_is_audio (uint32_t) const;
bool parameter_is_control (uint32_t) const;
bool parameter_is_input (uint32_t) const;
bool parameter_is_output (uint32_t) const;
-
- XMLNode& get_state();
- int set_state(const XMLNode& node);
-
- bool save_preset (string name);
- bool load_preset (const string preset_label);
- std::vector<std::string> get_presets ();
-
+
+ void set_info (PluginInfoPtr);
+
+ int set_state(const XMLNode& node, int);
+
+ bool load_preset (PresetRecord);
+ std::string current_preset() const;
+
bool has_editor () const;
-
- CAAudioUnit* get_au () { return unit; }
- CAComponent* get_comp () { return comp; }
-
+
+ bool can_support_io_configuration (const ChanCount& in, ChanCount& out, ChanCount* imprecise);
+ ChanCount output_streams() const;
+ ChanCount input_streams() const;
+ bool configure_io (ChanCount in, ChanCount out);
+ bool requires_fixed_size_buffers() const;
+
+ void set_fixed_size_buffers (bool yn) {
+ _requires_fixed_size_buffers = yn;
+ }
+
+ boost::shared_ptr<CAAudioUnit> get_au () { return unit; }
+ boost::shared_ptr<CAComponent> get_comp () const { return comp; }
+
+ OSStatus render_callback(AudioUnitRenderActionFlags *ioActionFlags,
+ const AudioTimeStamp *inTimeStamp,
+ UInt32 inBusNumber,
+ UInt32 inNumberSamples,
+ AudioBufferList* ioData);
+
+ /* "host" callbacks */
+
+ OSStatus get_beat_and_tempo_callback (Float64* outCurrentBeat,
+ Float64* outCurrentTempo);
+
+ OSStatus get_musical_time_location_callback (UInt32* outDeltaSampleOffsetToNextBeat,
+ Float32* outTimeSig_Numerator,
+ UInt32* outTimeSig_Denominator,
+ Float64* outCurrentMeasureDownBeat);
+
+ OSStatus get_transport_state_callback (Boolean* outIsPlaying,
+ Boolean* outTransportStateChanged,
+ Float64* outCurrentSampleInTimeLine,
+ Boolean* outIsCycling,
+ Float64* outCycleStartBeat,
+ Float64* outCycleEndBeat);
+
+ static std::string maybe_fix_broken_au_id (const std::string&);
+
+ /* this MUST be called from thread in which you want to receive notifications
+ about parameter changes.
+ */
+ int create_parameter_listener (AUEventListenerProc callback, void *arg, float interval_secs);
+ /* these can be called from any thread but SHOULD be called from the same thread
+ that will receive parameter change notifications.
+ */
+ int listen_to_parameter (uint32_t param_id);
+ int end_listen_to_parameter (uint32_t param_id);
+
+
+ protected:
+ std::string do_save_preset (std::string name);
+ void do_remove_preset (std::string);
+
private:
- CAComponent* comp;
- CAAudioUnit* unit;
+ void find_presets ();
+
+ boost::shared_ptr<CAComponent> comp;
+ boost::shared_ptr<CAAudioUnit> unit;
- AudioBufferList* in_list;
- AudioBufferList* out_list;
+ bool initialized;
+ int32_t input_channels;
+ int32_t output_channels;
+ std::vector<std::pair<int,int> > io_configs;
+ samplecnt_t _last_nframes;
+ mutable volatile guint _current_latency;
+ bool _requires_fixed_size_buffers;
+ AudioBufferList* buffers;
+ bool _has_midi_input;
+ bool _has_midi_output;
+ PluginOutputConfiguration _output_configs;
- std::vector<std::pair<uint32_t, uint32_t> > parameter_map;
+ /* despite all the cool work that apple did on their AU preset
+ system, they left factory presets and user presets as two
+ entirely different kinds of things, handled by two entirely
+ different parts of the API. Resolve this.
+ */
+
+ /* XXX these two maps should really be shared across all instances of this AUPlugin */
+
+ typedef std::map<std::string,std::string> UserPresetMap;
+ UserPresetMap user_preset_map;
+ typedef std::map<std::string,int> FactoryPresetMap;
+ FactoryPresetMap factory_preset_map;
+
+ UInt32 global_elements;
+ UInt32 output_elements;
+ UInt32 input_elements;
+
+ bool variable_inputs;
+ bool variable_outputs;
+
+ uint32_t configured_input_busses;
+ uint32_t configured_output_busses;
+
+ uint32_t *bus_inputs;
+ uint32_t *bus_outputs;
+ std::vector <std::string> _bus_name_in;
+ std::vector <std::string> _bus_name_out;
+
+ int set_stream_format (int scope, uint32_t bus, AudioStreamBasicDescription&);
+ void discover_parameters ();
+ void add_state (XMLNode *) const;
+
+ typedef std::map<uint32_t, uint32_t> ParameterMap;
+ ParameterMap parameter_map;
+ uint32_t input_maxbuf;
+ samplecnt_t input_offset;
+ samplecnt_t *cb_offsets;
+ BufferSet* input_buffers;
+ ChanMapping * input_map;
+ samplecnt_t samples_processed;
+ uint32_t audio_input_cnt;
+
+ std::vector<AUParameterDescriptor> descriptors;
+ AUEventListenerRef _parameter_listener;
+ void * _parameter_listener_arg;
+ void init ();
+
+ void discover_factory_presets ();
+
+ samplepos_t transport_sample;
+ float transport_speed;
+ float last_transport_speed;
+
+ static void _parameter_change_listener (void* /*arg*/, void* /*src*/, const AudioUnitEvent* event, UInt64 host_time, Float32 new_value);
+ void parameter_change_listener (void* /*arg*/, void* /*src*/, const AudioUnitEvent* event, UInt64 host_time, Float32 new_value);
};
typedef boost::shared_ptr<AUPlugin> AUPluginPtr;
-class AUPluginInfo : public PluginInfo {
- public:
- AUPluginInfo () { };
- ~AUPluginInfo ();
+struct LIBARDOUR_API AUPluginCachedInfo {
+ std::vector<std::pair<int,int> > io_configs;
+};
- CAComponentDescription* desc;
+class LIBARDOUR_API AUPluginInfo : public PluginInfo {
+ public:
+ AUPluginInfo (boost::shared_ptr<CAComponentDescription>);
+ ~AUPluginInfo ();
- static PluginInfoList discover ();
PluginPtr load (Session& session);
+ std::vector<Plugin::PresetRecord> get_presets (bool user_only) const;
+
+ bool needs_midi_input () const;
+ bool is_effect_without_midi_input () const;
+ bool is_effect_with_midi_input () const;
+
+ /* note: AU's have an explicit way to prompt for instrument/fx category */
+ bool is_effect () const;
+ bool is_instrument () const;
+ bool is_utility () const;
+
+ AUPluginCachedInfo cache;
+
+ bool reconfigurable_io() const { return true; }
+
+ static void clear_cache ();
+ static PluginInfoList* discover (bool scan_only);
+ static bool au_get_crashlog (std::string &msg);
+ static std::string stringify_descriptor (const CAComponentDescription&);
+
+ static int load_cached_info ();
+
private:
- static std::string get_name (CAComponentDescription&);
- void setup_nchannels (CAComponentDescription&);
+ boost::shared_ptr<CAComponentDescription> descriptor;
+ UInt32 version;
+ static FILE * _crashlog_fd;
+ static bool _scan_only;
+
+ static void au_start_crashlog (void);
+ static void au_remove_crashlog (void);
+ static void au_crashlog (std::string);
+
+ static void discover_music (PluginInfoList&);
+ static void discover_fx (PluginInfoList&);
+ static void discover_generators (PluginInfoList&);
+ static void discover_instruments (PluginInfoList&);
+ static void discover_by_description (PluginInfoList&, CAComponentDescription&);
+ static Glib::ustring au_cache_path ();
+
+ typedef std::map<std::string,AUPluginCachedInfo> CachedInfoMap;
+ static CachedInfoMap cached_info;
+
+ static int cached_io_configuration (const std::string&, UInt32, CAComponent&, AUPluginCachedInfo&, const std::string& name);
+ static void add_cached_info (const std::string&, AUPluginCachedInfo&);
+ static void save_cached_info ();
};
typedef boost::shared_ptr<AUPluginInfo> AUPluginInfoPtr;