This chapter covers connectors, connect-equations, and connections.
Connectors and connect-equations are designed so that different components can be connected graphically with well-defined semantics. However, the graphical part is optional and found in chapter 18.
Connections between objects are introduced by connect-equations in the equation part of a class. A connect-equation has the following syntax:
[A connector is an instance of a connector class.]
The connect-equation construct takes two references to connectors, each of which is either of the following forms:
m.c, where m is a non-connector element in the class and c is a connector element of m.
There may optionally be array subscripts on any of the components; the array subscripts shall be evaluable expressions or the special operator :. If the connect construct references array of connectors, the array dimensions must match, and each corresponding pair of elements from the arrays is connected as a pair of scalar connectors.
[Example: Array usage:
]
The three main tasks are to:
Elaborate expandable connectors.
Build connection sets from connect-equations.
Generate equations for the complete model (connection equations).
A connection set is a set of variables connected by means of connect-equations. A connection set shall contain either only flow variables or only non-flow variables.
In an element instance M, each connector element of M is called an outside connector with respect to M. Any other connector elements that is hierarchically inside M, but not in one of the outside connectors of M, is called an inside connector with respect to M. This is done before resolving outer elements to corresponding inner ones.
[Example:
The figure visualizes the following connect-equations to the connector c in the models m
and in the model for component m3 (c.x is a sub-connector inside c):
and in the model for component m6:
]
If the expandable qualifier is present on a connector definition, all instances of that connector are referred to as expandable connectors. Instances of connectors that do not possess this qualifier will be referred to as non-expandable connectors.
Before generating connection equations, non-parameter scalar variables and non-parameter array elements declared in expandable connectors are marked as only being potentially present. A non-parameter array element may be declared with array dimensions : indicating that the size is unknown (note that the size of such a dimension cannot be determined using size, see section 10.3.1). This applies to both variables of simple types, and variables of structured types.
Then connections containing expandable connectors are elaborated:
If a connect-equation references a potentially present component as part of the argument it will be marked as being present, which will allow a connection to an undeclared connector inside it. The rule does not apply for the complete argument.
After that at least one connector in the connect-equation must reference a declared component.
If the other connector is undeclared it must be in a declared component and is handled as follows:
The expandable connector instance is automatically augmented with a new component having the used name and corresponding type.
If the undeclared component is subscripted, an array variable is created, and a connection to the specific array element is performed. Introducing elements in an array gives an array with at least the specified elements, other elements are either not created or have a default value (i.e., as if they were only potentially present, and the same note regarding the use of size also applies here).
If the variable on the other side of the connect-equation is input or output the new component will be either input or output to satisfy the restrictions in section 9.3 for a non-expandable connector.
[The general rule ensures consistency for inside and outside connectors, and handles multiple connections to the new component. In the simple case of no other connections involving these variables and the existing side referring to an inside connector (i.e., a connector of a component), the new variable will copy its causality (i.e., input if input and output if output) since the expandable connector must be an outside connector.]
For an array the input/output property can be deduced separately for each array element.
Additionally:
When two expandable connectors are connected, each is augmented with the variables that are only declared in the other expandable connector (the new variables are neither input nor output). This is repeated until all connected expandable connector instances have matching variables.
[I.e., each of the connector instances is expanded to be the union of all connector variables.]
The variables introduced in the elaboration follow additional rules for generating connection sets (given in section 9.2).
If a variable appears as an input in one expandable connector, it should appear as a non-input in at least one other expandable connector instance in the same augmentation set. An augmentation set is defined as the set of connected expandable connector instances that through the elaboration will have matching variables.
[Example:
This small example shows how expandable connectors are normally used:
There are a number of bus-instances all connected together. Often they have the same name, but it is not required.
There is one source of the signal: sensor.sensor.speed.
There are zero or more uses of the signal: actuator.actuator.speed.
]
All components in an expandable connector are seen as connector instances even if they are not declared as such.
[I.e., it is possible to connect to, e.g., a Real variable.]
[Example:
]
An expandable connector shall not contain a component declared with the prefix flow, but may contain non-expandable connector components with flow components.
[Example:
]
Expandable connectors can only be connected to other expandable connectors.
If a connect-equation references a potentially present variable, or variable element, in an expandable connector the variable or variable element is marked as being present, and due to the paragraphs above it is possible to deduce whether the bus variable shall be treated as input, or shall be treated as output in the connect-equation. That input or output prefix is added if no input/output prefix is present on the declaration.
[Example:
]
An expandable connector array component for which size is not defined (see section 10.3.1) is referred to as a sizeless array component. Such a component shall not be used without subscripts, and the subscripts must slice the array so that the sizeless dimensions are removed.
[Example: Valid and invalid uses of sizeless array components:
]
After this elaboration the expandable connectors are treated as normal connector instances, and the connections as normal connections, and all potentially present variables and array elements that are not actually present are undefined. It is an error if there are expressions referring to potentially present variables or array elements that are not actually present or non-declared variables. This elaboration implies that expandable connectors can be connected even if they do not contain the same components.
[A tool may remove undefined variables in an expandable connector, or set them to the default value, e.g., zero for Real variables.]
[Expressions can only “read” variables from the bus that are actually declared and present in the connector, in order that the types of the variables can be determined in the local scope.]
[Note that the introduction of variables, as described above, is conceptual and does not necessarily impact the flattening hierarchy in any way. Furthermore, it is important to note that these elaboration rules must consider:
Expandable connectors nested hierarchically. This means that both outside and inside connectors must be included at every level of the hierarchy in this elaboration process.
When processing an expandable connector that possesses the inner scope qualifier, all outer instances must also be taken into account during elaboration.
]
[Example: Engine system with sensors, controllers, actuator and plant that exchange information via a bus (i.e., via expandable connectors):
Due to the above connection, conceptually a connector consisting of the union of all connectors is introduced.
The engine_bus contains the following variable declarations:
]
When generating connection equations, outer elements are resolved to the corresponding inner elements in the instance hierarchy (see instance hierarchy name lookup section 5.4). The arguments to each connect-equation are resolved to two connector elements.
For every use of the connect-equation
a connection set is generated for each pair of corresponding primitive components of a and b together with an indication of whether they are from an inside or an outside connector.
The primitive elements are of simple types or of types defined as operator record (i.e., a component of an operator record type is not split into sub-components). ∎
The elements of the connection sets are tuples of primitive variables together with an indication of inside or outside; if the same tuple belongs to two connection sets those two sets are merged, until every tuple is only present in one set. Composite connector types are broken down into primitive components. The outer components are handled by mapping the objects to the corresponding inner components, and the inside indication is not influenced. The outer connectors are handled by mapping the objects to the corresponding inner connectors, and they are always treated as outside connectors.
[Rationale: The inside/outside as part of the connection sets ensure that connections from different hierarchical levels are treated separately. Connection sets are formed from the primitive elements and not from the connectors; this handles connections to parts of hierarchical connectors and also makes it easier to generate equations directly from the connection sets. All variables in one connection set will either be flow variables or non-flow variables due to restriction on connect-equations. The mapping from an outer to an inner element must occur before merging the sets in order to get one zero-sum equation, and ensures that the equations for the outer elements are all given for one side of the connector, and the inner element can define the other side.]
The following connection sets with just one member are also present (and merged):
Each primitive flow variable as inside connector.
Each flow variable added during augmentation of expandable connectors, both as inside and as outside.
[Note that the flow variable is not directly in the expandable connector, but in a connector inside the expandable connector.]
[Rationale: If these variables are not connected they will generate a set comprised only of this element, and thus they will be implicitly set to zero (see below). If connected, this set will be merged and adding this at the start has no impact.]
Each connection set is used to generate equations for potential and flow (zero-sum) variables of the form
The bold-face
For an operator record type this uses the operator '0' – which must be defined in the operator record – and all of the flow variables for the operator record must be of the same operator record type. This implies that in order to have flow variables of an operator record type the operator record must define addition, negation, and '0'; and these operations should define an additive group.
In order to generate equations for flow variables (using the flow prefix), the sign used for the connector variable
[Example: Simple example:
The connection sets are before merging (note that one part of the load and resistor is not connected):
{<load.p.i, inside>}
{<load.n.i, inside>}
{<ground.p.i, inside>}
{<load.resistor.p.i, inside>}
{<load.resistor.n.i, inside>}
{<resistor.p.i, inside>}
{<resistor.n.i, inside>}
{<resistor.p.i, inside>, <ground.p.i, inside>}
{<resistor.p.v, inside>, <ground.p.v, inside>}
{<load.p.i, inside>, <ground.p.i, inside>}
{<load.p.v, inside>, <ground.p.v, inside>}
{<load.p.i, outside>, <load.resistor.p.i, inside>}
{<load.p.v, outside>, <load.resistor.p.v, inside>}
{<load.n.i, outside>, <load.resistor.n.i, inside>}
{<load.n.v, outside>, <load.resistor.n.v, inside>}
After merging this gives:
{<load.p.i, outside>, <load.resistor.p.i, inside>}
{<load.p.v, outside>, <load.resistor.p.v, inside>}
{<load.n.i, outside>, <load.resistor.n.i, inside>}
{<load.n.v, outside>, <load.resistor.n.v, inside>}
{<load.p.i, inside>, <ground.p.i, inside>, <resistor.p.i, inside>}
{<load.p.v, inside>, <ground.p.v, inside>, <resistor.p.v, inside>}
{<load.n.i, inside>}
{<resistor.n.i, inside>}
And thus the equations:
]
[Example: Outer component example:
The connection sets are before merging (note that one part of the load and resistor is not connected):
{<load.p.i, inside>}
{<load.n.i, inside>}
{<ground.p.i, inside>}
{<resistor.p.i, inside>}
{<resistor.n.i, inside>}
{<load.p.i, inside>, <ground.p.i, inside>}
{<load.p.v, inside>, <ground.p.v, inside>}
{<load.p.i, outside>, <resistor.p.i, inside>}
{<load.p.v, outside>, <resistor.p.v, inside>}
{<load.n.i, outside>, <resistor.n.i, inside>}
{<load.n.v, outside>, <resistor.n.v, inside>}
After merging this gives:
{<load.p.i, outside>, <resistor.p.i, inside>}
{<load.p.v, outside>, <resistor.p.v, inside>}
{<load.n.i, outside>, <resistor.n.i, inside>}
{<load.n.v, outside>, <resistor.n.v, inside>}
{<load.p.i, inside>, <ground.p.i, inside>}
{<load.p.v, inside>, <ground.p.v, inside>}
{<load.n.i, inside>}
And thus the equations:
This corresponds to a direct connection of the resistor.]
The connect-equations (and the special functions for overdetermined connectors) can only be used in equations, and shall not be used inside if-equations with conditions that are not evaluable expressions, or in when-equations.
[The for-equations always have evaluable expressions for the array expression.]
A connector component shall not be declared with the prefix parameter or constant. In the connect-equation the primitive components may only connect parameter variables to parameter variables and constant variables to constant variables.
The connect-equation construct only accepts forms of connector references as specified in section 9.1.
In a connect-equation the two connectors must have the same named component elements with the same dimensions; recursively down to the primitive components. The primitive components with the same name are matched and belong to the same connection set.
The matched primitive components of the two connectors must have the same primitive types, and flow variables may only connect to other flow variables, stream variables only to other stream variables, and causal variables (input/output) only to causal variables (input/output).
A connection set of causal variables (input/output) may at most contain variables from one inside output connector (for state-machines extended as specified in section 17.3.6) or one public outside input connector.
[I.e., a connection set may at most contain one source of a signal.]
At least one of the following must hold for a connection set containing causal variables generated for a non-partial model or block:
the connection set includes variables from an outside public expandable connector,
the set contains variables from protected outside connectors,
it contains variables from one inside output connector, or
from one public outside input connector, or
the set is comprised solely of one variable from one inside input connector that is not part of an expandable connector.
[I.e., a connection set must – unless the model or block is partial – contain one source of a signal (item 5 covers the case where a connector of a component is left unconnected and the source given textually).]
Variables from a protected outside connector must be part of a connection set containing at least one inside connector or one declared public outside connector (i.e., it shall not be an implicitly defined part of an expandable connector).
[Otherwise it would not be possible to deduce the causality for the expandable connector element.]
In a connection set, all variables having non-empty quantity-attribute must have the same quantity-attribute.
A connect-equation shall not (directly or indirectly) connect two connectors of outer elements.
[Indirectly is similar to them being part of the same connection set. However, connections to outer elements are “moved up” before forming connection sets. Otherwise the connection sets could contain redundant information breaking the equation count for locally balanced models and blocks.]
Subscripts in a connector reference shall be evaluable expressions or the special operator :.
Constants or parameters in connected components yield the appropriate assert-statements to check that they have the same value; connections are not generated.
For conditional connectors, see section 4.4.5.
For each non-partial non-simple non-expandable connector class the number of flow variables shall be equal to the number of variables that are neither parameter, constant, input, output, stream nor flow. The number of variables is the number of all elements in the connector class after expanding all records and arrays to a set of scalars of primitive types. The number of variables of an overdetermined type or record class (see section 9.4.1) is the size of the output argument of the corresponding equalityConstraint() function. A simple connector class is not expandable, has some time-varying variables, and has neither input, output, stream nor flow variables.
[Expandable connector classes are excluded from this, since their component declarations are only a form of constraint.]
A component of a simple connector class must be declared as input, output, or protected.
[A simple connector class is thus always unbalanced, but since it is used with causality or not visible to the outside it does not unbalance the model.]
[Example:
The Frame_Illegal connector (intended to be used in a simple multi-body package without over-determined connectors) is illegal since the number of flow and non-flow variables do not match. The solution is to create two connector classes, where two 3-vectors (e.g., a and z) are acausal Real and the other variables are matching pairs of input and output. This ensures that the models can only be connected in a tree-structure or require a “loop-breaker” joint for every closed kinematic loop:
The subsequent connectors Plug_Expanded and PlugExpanded2 are correct, but Plug_Expanded_Illegal is illegal since the number of non-flow and flow variables is different if n and m are different. It is not clear how a tool can detect in general that connectors such as Plug_Expanded_Illegal are illegal. However, it is always possible to detect this defect after actual values of parameters and constants are provided in the simulation model.
]
There is a special problem regarding equation systems resulting from loops in connection graphs where the connectors contain non-flow (i.e., potential) variables dependent on each other. When a loop structure occurs in such a graph, the resulting equation system will be overconstrained, i.e., have more equations than variables, since there are implicit constraints between certain non-flow variables in the connector in addition to the connection equations around the loop. At the current state-of-the-art, it is not possible to automatically eliminate the unneeded equations from the resulting equation system without additional information from the model designer.
This section describes a set of equation operators for such overconstrained connection-based equation systems, that makes it possible for the model designer to specify enough information in the model to allow a Modelica environment to automatically remove the superfluous equations.
[Connectors may contain redundant variables.
For example, the orientation between two coordinate systems in
This section handles a certain class of overdetermined systems due to connectors that have a redundant set of variables. There are other causes of overdetermined systems, e.g., explicit zero-sum equations for flow variables, that are not handled by the method described below.]
A type or record declaration may have an optional definition of function equalityConstraint that shall have the following prototype:
The array dimension
[If the elements of a record Record are not independent from each other, the equation R1 = R2 contains redundant equations.]
A type class with an equalityConstraint function declaration is called overdetermined type. A record class with an equalityConstraint function definition is called overdetermined record. A connector that contains instances of overdetermined type and/or record classes is called overdetermined connector. An overdetermined type or record may neither have flow components nor may be used as a type of flow components. If an array is used as argument to any of the Connections.* functions it is treated as one unit – unlike connect, there is no special treatment of this case, compare section 9.1.
Every instance of an overdetermined type or record in an overdetermined connector is a node in a virtual connection graph that is used to determine when the standard equation R1 = R2 or when the equation 0 = equalityConstraint(R1, R2) has to be used for the generation of connect-equations. The edges of the virtual connection graph are implicitly defined by connect and explicitly by Connections.branch, see table below. Connections is a built-in package in global scope containing built-in operators. Additionally, corresponding nodes of the virtual connection graph have to be defined as roots or as potential roots with functions Connections.root and Connections.potentialRoot, respectively.
The overconstrained equation operators for connection graphs are listed below. Here, a and b are connector instances that may be hierarchically structured, e.g., a may be an abbreviation for enginePort.frame_a.
Expression | Description | Details |
---|---|---|
connect(a, b) | Optional spanning-tree edge | Operator 9.1 |
Connections.branch(a.R, b.R) | Required spanning-tree edge | Operator 9.2 |
Connections.root(a.R) | Definite root node | Operator 9.3 |
Connections.potentialRoot(a.R, |
Potential root node | Operator 9.4 |
Connections.isRoot(a.R) | Predicate for being selected as root | Operator 9.5 |
Connections.rooted(a.R) | Predicate for being closer to root | Operator 9.6 |
Except for redundant connections it defines an optional spanning-tree edge from the overdetermined type or record instances in connector a to the corresponding overdetermined type or record instances in connector b for a virtual connection graph. E.g., from a.R to b.R. The full explanation will be given in section 9.4.2. The types of the corresponding overdetermined type or record instances shall be the same.
Defines a required spanning-tree edge from the overdetermined type or record instance R in connector instance a to the corresponding overdetermined type or record instance R in connector instance b for a virtual connection graph. This function can be used at all places where a connect-equation is allowed.
[E.g., it is not allowed to use this function in a when-clause.
This definition shall be used if in a model with connectors a and b the overdetermined records a.R and b.R are algebraically coupled in the model, e.g., due to b.R = f(a.R,
The overdetermined type or record instance R in connector instance a is a (definite) root node in a virtual connection graph.
[This definition shall be used if in a model with connector a the overdetermined record a.R is (consistently) assigned, e.g., from a parameter expressions.]
The overdetermined type or record instance R in connector instance a is a potential root node in a virtual connection graph with priority p (
[This definition may be used if in a model with connector a the overdetermined record a.R appears differentiated – der(a.R) – together with the constraint equations of a.R, i.e., a non-redundant subset of a.R maybe used as states.]
Returns true, if the overdetermined type or record instance R in connector instance a is selected as a root in the virtual connection graph.
If the operator Connections.rooted(a.R) is used, or the equivalent but deprecated operator rooted(a.R), then there must be exactly one Connections.branch(a.R, b.R) involving a.R (the argument of Connections.rooted must be the first argument of Connections.branch). In that case Connections.rooted(a.R) returns true, if a.R is closer to the root of the spanning tree than b.R; otherwise false is returned.
[This operator can be used to avoid equation systems by providing analytic inverses, see Modelica.Mechanics.MultiBody.Parts.FixedRotation.]
[Note, that Connections.branch, Connections.root, Connections.potentialRoot do not generate equations. They only generate nodes and edges in the virtual graph for analysis purposes.]
When generating connection graph equations, only the overdetermined components of a connector are considered. The connection graph equations replace the equality-equations for variables that are neither flow nor stream in section 9.2.
In order to eliminate any redundant connect-equation the following preparation is needed.
[In the common case where there is no connect-equation redundancy, a consequence of this preparation is that a connect-equation between connectors with one overdetermined component may be directly turned into one optional spanning-tree edge.]
The connection sets are built similarly to the normal way, but keeping the overdetermined components as primitives.
Instead of generating the equality-equation for an overdetermined component, an optional spanning-tree edge in the virtual connection graph is constructed.
[If a connection set contains
It is called redundancy since this number of connect-equations could be removed without changing the connection set or the generated equations. Situations with non-zero connect-equation redundancy include connectors connected directly to themselves, duplicated connections, and having all three pair-wise connections between the connectors a, b and c. The latter case can be used to consistently handle conditional components (so that disabling b does not break the connection between a and c).]
The selected optional spanning tree edges, together with all required spanning tree edges generated from Connections.branch, and nodes corresponding to definite and potential roots form the virtual connection graph.
Before connection equations are generated, the virtual connection graph is transformed into a set of spanning trees by removing optional spanning tree edges from the graph. This is performed in the following way:
Root nodes are selected as follows:
Every definite root node defined via the Connections.root-equation is a root of one spanning tree. It is an error if two (or more) definite root nodes are connected through required spanning tree edges.
The virtual connection graph may consist of sets of subgraphs that are not connected together. Every subgraph in this set shall have at least one definite root node or one potential root node in a simulation model. If a graph of this set does not contain any definite root node, then one potential root node in this subgraph that has the lowest priority number is selected to be the root of that subgraph. The selection can be inquired in a class with function Connections.isRoot, see table above.
If there is a cycle among required spanning-tree-edges it is an error, as it is not possible to construct a spanning tree.
For a subgraph with
After this analysis, the connection graph equations are generated in the following way:
For every remaining optional spanning-tree edge in any of the spanning trees, the connection equations are generated according to section 9.2. For connect(a, b) with overdetermined connector R, this corresponds to the optional spanning-tree edge between a.R and b.R generating the equation a.R = b.R.
For every remaining optional spanning-tree edge not in any of the spanning trees, the connection equations are generated according to section 9.2, except for overdetermined type or record instances R. Here the equations 0 = R.equalityConstraint(a.R, b.R) are generated instead of a.R = b.R.
[Example:
]
[An overdetermined connector for power systems based on the transformation theory of Park may be defined as:
The currents and voltages in the connector are defined relatively to the harmonic, high-frequency signal of a power source that is essentially described by angle theta of the rotor of the source. This allows much faster simulations, since the basic high frequency signal of the power source is not part of the differential equations. For example, when the source and the rest of the line operates with constant frequency (= nominal case), then AC_Plug.v and AC_Plug.i are constant. In this case a variable step integrator can select large time steps. An element, such as a 3-phase inductor, may be implemented as:
At the place where the source frequency, i.e., essentially variable theta, is defined, a Connections.root must be present:
The graph analysis performed with the virtual connection graph identifies the connectors, where the AC_Angle needs not to be passed between components, in order to avoid redundant equations.
Note that the different sources do not integrate the frequency, as that increases the risk of numerical errors.]
[An overdetermined connector for 3-dimensional mechanical systems may be defined as:
A fixed translation from a frame a to a frame b may be defined as:
Since the transformation matrix frame_a.R is algebraically coupled with frame_b.R, an edge in the virtual connection graph has to be defined. At the inertial system, the orientation is consistently initialized and therefore the orientation in the inertial system connector has to be defined as root:
]