refine self-automating plugin interface
[ardour.git] / libs / ardour / ardour / chan_mapping.h
index 1dae20e34aa439c9c40dc9402b0d0c7536c85888..ac546e3bbd6000b180ec22db4a20b1c600d307bb 100644 (file)
@@ -1,6 +1,6 @@
 /*
-    Copyright (C) 2009 Paul Davis 
-       Author: Dave Robillard
+    Copyright (C) 2009 Paul Davis
+    Author: David Robillard
 
     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
 #include <ostream>
 #include <utility>
 
+#include "pbd/xml++.h"
 #include "ardour/data_type.h"
+#include "ardour/chan_count.h"
 
 namespace ARDOUR {
 
 
-/** A mapping from one set of channels to another
- * (e.g. how to 'connect' two BufferSets).
+/** A mapping from one set of channels to another.
+ * The general form is  1 source (from), many sinks (to).
+ * numeric IDs are used to identify sources and sinks.
+ *
+ * for plugins this is used to map "plugin-pin" to "audio-buffer"
  */
-class ChanMapping {
+class LIBARDOUR_API ChanMapping {
 public:
        ChanMapping() {}
-       ChanMapping(ChanCount identity) {
-               for (DataType::iterator t = DataType::begin(); t != DataType::end(); ++t) {
-                       for (size_t i = 0; i <= identity.get(*t); ++i)
-                               set(*t, i, i);
-               }
-       }
+       ChanMapping(ARDOUR::ChanCount identity);
+       ChanMapping(const ChanMapping&);
+       ChanMapping(const XMLNode& node);
 
-       uint32_t get(DataType t, uint32_t from) {
-               Mappings::iterator tm = _mappings.find(t);
-               assert(tm != _mappings.end());
-               TypeMapping::iterator m = tm->second.find(from);
-               assert(m != tm->second.end());
-               return m->second;
-       }
-       
-       void set(DataType t, uint32_t from, uint32_t to) {
-               Mappings::iterator tm = _mappings.find(t);
-               if (tm == _mappings.end()) {
-                       tm = _mappings.insert(std::make_pair(t, TypeMapping())).first;
-               }
-               tm->second.insert(std::make_pair(from, to));
+       uint32_t get(DataType t, uint32_t from, bool* valid) const;
+
+       /** reverse lookup
+        * @param type data type
+        * @param to pin
+        * @param valid pointer to a boolean. If not NULL it is set to true if the mapping is found, and false otherwise.
+        * @returns first "from" that matches given "to"
+        */
+       uint32_t get_src(DataType t, uint32_t to, bool* valid) const;
+
+       /** get buffer mapping for given data type and pin
+        * @param type data type
+        * @param from numeric source id
+        * @returns mapped buffer number (or ChanMapping::Invalid)
+        */
+       uint32_t get (DataType t, uint32_t from) const { return get (t, from, NULL); }
+
+       /** set buffer mapping for given data type
+        * @param type data type
+        * @param from numeric source id
+        * @param to buffer
+        */
+       void     set(DataType t, uint32_t from, uint32_t to);
+       void     offset_from(DataType t, int32_t delta);
+       void     offset_to(DataType t, int32_t delta);
+
+       /** remove mapping
+        * @param type data type
+        * @param from numeric source to remove from mapping
+        */
+       void     unset(DataType t, uint32_t from);
+
+       /** Test mapping matrix for identity
+        * @param offset per data-type offset to take into account
+        * @returns true if the mapping is a channel identity map
+        */
+       bool     is_identity (ARDOUR::ChanCount offset = ARDOUR::ChanCount()) const;
+
+       /** Test if this mapping is monotonic (useful to see if inplace processing is feasible)
+        * @returns true if the map is a strict monotonic set
+        */
+       bool     is_monotonic () const;
+
+       uint32_t n_total () const;
+
+       ChanCount count () const;
+
+       XMLNode* state(const std::string& name) const;
+
+       /** Test if this mapping is a subset
+        * @param superset to test against
+        * @returns true if all mapping are also present in the superset
+        */
+       bool     is_subset (const ChanMapping& superset) const;
+
+       typedef std::map<uint32_t, uint32_t>    TypeMapping;
+       typedef std::map<DataType, TypeMapping> Mappings;
+
+       Mappings       mappings()       { return _mappings; }
+       const Mappings mappings() const { return _mappings; }
+
+       bool operator==(const ChanMapping& other) const {
+               return (_mappings == other._mappings);
        }
 
-       /** Increase the 'to' field of every mapping for type @a t by @a delta */
-       void offset(DataType t, uint32_t delta) {
-               Mappings::iterator tm = _mappings.find(t);
-               if (tm != _mappings.end()) {
-                       for (TypeMapping::iterator m = tm->second.begin(); m != tm->second.end(); ++m) {
-                               m->second += delta;
-                       }
-               }
+       bool operator!=(const ChanMapping& other) const {
+               return ! (*this == other);
        }
 
 private:
-       typedef std::map<uint32_t, uint32_t>    TypeMapping;
-       typedef std::map<DataType, TypeMapping> Mappings;
-       
        Mappings _mappings;
 };
 
 } // namespace ARDOUR
 
+std::ostream& operator<<(std::ostream& o, const ARDOUR::ChanMapping& m);
+
 #endif // __ardour_chan_mapping_h__