A modular synthesizer usually contains modules which have similarities. Oscillator modules for example have a controller for it's frequency and an (audio) output connector. The JPatch API supports taking advantage of these similarities. Mappings from one module to another can be defined in a XML file and the transformations can be automated. The user can use this feature to switch from one module to another while (some of the) parameter values and connections are retained.
In this document the term mapping refers to the transformation t := (ma → mb) and the inverse t-1 := (mb → ma) of the modules ma, module mb.
Each mapping contains the components (parameters and connectors) of two modules which can be mapped. The components have a selector id associated with them which is used to link a component of module ma and a component of module mb.
Example:
Sine Osc. | selector id | mapping | selector id | Pulse Osc. |
---|---|---|---|---|
parameter: frequency | freq | freq | freq | parameter: frequency |
parameter: fine tune | finetune | - | pwma | parameter: pulse width mod amount |
connector: audio out | aout | aout | aout | connector: audio out |
- | - | - | fm | connector: fm in |
- | - | - | pwm | connector: pulse width mod |
⋮ | ⋮ | ⋮ | ⋮ | ⋮ |
Using this table the mapping is defined as the intersection of the selectors: mapping(ma,mb) := selectors(ma) ∩ selectors(mb)
Each pair of modules (ma,mb) defines a possible mapping, thus the mappings can be defined as a matrix with a row/column for each module. Because the the pairs of modules (ma,mb) and (mb,ma) refer to the same mapping the matrix is symmetrical. The pairs (ma, ma) are the identical mapping (diagonale).
m1 | m2 | ⋯ | mn | |
m1 | id | x | i | |
m2 | - | id | x | |
⋮ | ⋮ | ⋱ | ⋮ | |
mn | - | - | ⋯ | id |
If an initial set of mappings is defined it is possible to add some of the missing mappings to the matrix using an algorithm. For example in the matrix above the mappings (m1, m2) and (m2, mn) were defined at the beginning. These two mappings imply the third mapping (m1, mn). However, the algorithm must work carefully because different paths could result in better or worse mappings.
The initial mappings can be defined using the Transformations XML v1.1 format.
The element transformations
is the root element. It has an attribute version
containing the version
of the specification (current version "1.1"). The element contains any number of group
-elements.
<?xml version="1.0" encoding="utf-8"?> <transformations version="1.1"> ... </transformations>
The element group
has no attributes and contains any number module
-elements.
Each combination of the containing modules defines a mapping. Two different groups must not contain
the same pair of modules, more formally for each two groups g1 and g2 the condition
g1 ≠ g2 ⇒ | g1 ∩ g2 | < 2
must be true.
A group
can contain at most one module with a specific id (component-id
).
<group> ... </group>
The element module
has the attribute component-id
which contains the identifier (id) of the
module.
Possible child elements are parameter
and connector
.
<module component-id="SineOsc"> ... </module>
The element parameter
has the attribute component-id
which contains the id of the
parameter. The attribute selector
is used to map this parameter to another parameter in the same group.
A parameter (identified by it's component id) can be declared one time at most in the module
element
and there must not be any other parameter
or connector
element with the same selector
value.
<parameter component-id="frequency" selector="freq" />
The element connector
has the attribute component-id
which contains the id of the
connector. The attribute selector
is used to map this connector to another connector in the same group.
A connector (identified by it's component id) can be declared one time at most in the module
element
and there must not be any other parameter
or connector
element with the same selector
value.
The connectors which are mapped to each other must be of the same type either input or output.
<connector component-id="aout" selector="out" />
<?xml version="1.0" encoding="utf-8"?> <transformations version="1.1"> <group> <module component-id="SineOsc"> <parameter component-id="frequency" selector="freq" /> <connector component-id="audioout" selector="aout" /> </module> <module component-id="PulseOsc"> <parameter component-id="frequency" selector="freq" /> <connector component-id="audioout" selector="aout" /> <connector component-id="fm-in" selector="fmin" /> </module> <module component-id="SawtoothOsc"> <parameter component-id="frequency" selector="freq" /> <connector component-id="audioout" selector="aout" /> <connector component-id="fm-in" selector="fmin" /> </module> </group> <group> <module component-id="PulseOsc"> <parameter component-id="frequency" selector="freq" /> <connector component-id="audioout" selector="aout" /> <connector component-id="fm-in" selector="fmin" /> </module> <module component-id="PulseOscB"> <parameter component-id="frequency" selector="freq" /> <connector component-id="audioout" selector="aout" /> <connector component-id="fm-in" selector="fmin" /> </module> </group> <group> <module component-id="WaveWrapper"> <parameter component-id="amount" selector="amount" /> <connector component-id="audioout" selector="aout" /> </module> <module component-id="Overdrive"> <parameter component-id="overdrive" selector="amount" /> <connector component-id="audioout" selector="aout" /> </module> </group> </transformations>
The example document defines the initial mappings of the modules SinceOsc, PulseOsc, SawtoothOsc, PulseOscB, WaveWrapper, Overdrive. Three mappings are defined in the first group { (SineOsc, PulseOsc), (SinceOsc, SawtoothOsc), (PulseOsc,SawtoothOsc) }, one in the second group { (PulseOsc, PulseOscB) }, and one in the third group { (WaveWrapper, Overdrive) }.
SineOsc | PulseOsc | SawtoothOsc | PulseOscB | WaveWrapper | Overdrive | |
SineOsc | id | x | x | i | ||
PulseOsc | - | id | x | x | ||
SawtoothOsc | - | - | id | i | ||
PulseOscB | - | - | - | id | ||
WaveWrapper | - | - | - | - | id | x |
Overdrive | - | - | - | - | id |
Some of the undefined mappings are derived using the matrix.
The third group {WaveWrapper, Overdrive} is independent of the other two groups, no further mappings can be derived.
This is the graph of the matrix above. Each edge represents a mapping. The blue edges are defined in the XML file, the red edges are derived.