summaryrefslogtreecommitdiffstats
path: root/flow/virtualports.cpp
diff options
context:
space:
mode:
Diffstat (limited to 'flow/virtualports.cpp')
-rw-r--r--flow/virtualports.cpp491
1 files changed, 491 insertions, 0 deletions
diff --git a/flow/virtualports.cpp b/flow/virtualports.cpp
new file mode 100644
index 0000000..b733256
--- /dev/null
+++ b/flow/virtualports.cpp
@@ -0,0 +1,491 @@
+ /*
+
+ Copyright (C) 2000 Stefan Westerfeld
+
+ This library is free software; you can redistribute it and/or
+ modify it under the terms of the GNU Library General Public
+ License as published by the Free Software Foundation; either
+ version 2 of the License, or (at your option) any later version.
+
+ This library is distributed in the hope that it will be useful,
+ but WITHOUT ANY WARRANTY; without even the implied warranty of
+ MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
+ Library General Public License for more details.
+
+ You should have received a copy of the GNU Library General Public License
+ along with this library; see the file COPYING.LIB. If not, write to
+ the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
+ Boston, MA 02110-1301, USA.
+
+ */
+
+#include "virtualports.h"
+#include <algorithm>
+#include <stdio.h>
+
+using namespace Arts;
+using namespace std;
+
+#undef VPORT_DEBUG
+
+/* virtual port connections */
+
+/*
+
+Port virtualization is used in the following case: suppose you have a module
+M which has an input port and an output port like that:
+
+input
+ V
+-----
+ M
+-----
+ V
+output
+
+But suppose the module M is implement as combination of other modules, for
+instance the effect of M is achieved by passing the signal first through an
+A and then through an B module. Clients connecting M should "really" connect
+A,B. For this virtualization is used.
+
+There are two kinds:
+
+- masquerading: which means (like in our case), the input of a module is really
+ implemented with the input of another module. So M's input could be really
+ implemented by A's input
+
+ there is also output masquerading, which would be for instance that M's
+ output is really implemented by B's output
+
+- forwarding: if M in our example would choose to do nothing at all, it could
+ simply forward its input to its output
+
+
+The interface for the user:
+
+MCOP will show the virtualize function to the user, which the user can use
+to delegate the services of a port to another module onto another port.
+
+- masquerading: in our case, for instance the user could call
+
+ m._node()->virtualize("inputport",a._node(),"inputport");
+
+ which would forward all input m gets on "inputport" to a's "inputport"
+
+- forwarding: in the same way, the user could call
+
+ m._node()->virtualize("inputport",m._node(),"outputport");
+
+ which would make m forward its input directly to its output
+
+The implementation:
+
+Virtualization is implemented here, inside the flow system, using a fairly
+complex forwarding strategy, which will have a graph which contains
+
+- "user-made" connections (made with connect())
+- "virtualize-made" connections, which can be either forwarding
+ (input to output port) or masquerading (input to input or output to output)
+
+Out of all these, the algorithm builds "real" connections, which are
+then really performed inside the flow system. If you change the "user-made"
+or "virtualize-made" connections, the "real" connections are recalculated.
+
+The "real" connections are created by the expandHelper function. They are
+called vcTransport here.
+
+The strategy expandHelper uses is to go back to a port which is only output
+port (non forwarded, non masqueraded), and then follow the graph recursively
+over vcMasquerade and vcForward edges until it reaches a port which is only
+input. Then it creates a real connection.
+
+Some tweaks are there which allow that not on any change at the graph, all
+real connections will be removed, but only these that could possibly be
+affected by this change, and then not all real connections are created new,
+but only those that could possibly be created by this virtual connection.
+
+Every VPort contains a pointer to the "real" port, to let the flow system
+know where the "real" connections where real data will flow must be made.
+
+*/
+
+VPortConnection::VPortConnection(VPort *source, VPort *dest, Style style)
+ :source(source),dest(dest),style(style)
+{
+ if(style != vcTransport)
+ {
+ list<VPortConnection *>::iterator i;
+
+ // remove transport connections ending at "source" (they will
+ // probably be forwarded/masqueraded elsewhere by this VPortConnection)
+ i = source->incoming.begin();
+ while(i != source->incoming.end())
+ {
+ if((*i)->style == vcTransport)
+ {
+ delete *i;
+ i = source->incoming.begin();
+ }
+ else i++;
+ }
+
+ // remove transport connections starting at "dest" (they will
+ // probably be forwarded/masqueraded elsewhere by this VPortConnection)
+ i = dest->outgoing.begin();
+ while(i != dest->outgoing.end())
+ {
+ if((*i)->style == vcTransport)
+ {
+ delete *i;
+ i = dest->outgoing.begin();
+ }
+ else i++;
+ }
+ }
+
+ // add to the connection lists
+ source->outgoing.push_back(this);
+ dest->incoming.push_back(this);
+
+ if(style == vcTransport)
+ {
+#ifdef VPORT_DEBUG
+ arts_debug("emit a connection consumer = %s, producer = %s",
+ dest->name(), source->name());
+#endif
+ dest->port->connect(source->port);
+ }
+ else
+ {
+ source->makeTransport(this);
+ }
+}
+
+VPortConnection::~VPortConnection()
+{
+#ifdef VPORT_DEBUG
+ cout << "~VPortConnection" << endl;
+#endif
+
+ if(style != vcTransport)
+ {
+ // remove transport connection which go through this connection
+ source->removeTransport(this);
+ }
+
+ // remove this connection from the lists
+ list<VPortConnection *>::iterator ci;
+
+ ci = find(source->outgoing.begin(),source->outgoing.end(),this);
+ assert(ci != source->outgoing.end());
+ source->outgoing.erase(ci);
+
+ ci = find(dest->incoming.begin(),dest->incoming.end(),this);
+ assert(ci != dest->incoming.end());
+ dest->incoming.erase(ci);
+
+ if(style == vcTransport)
+ {
+#ifdef VPORT_DEBUG
+ arts_debug("delete connection %s -> %s",dest->name(), source->name());
+#endif
+ dest->port->disconnect(source->port);
+ }
+
+ // reestablish all connections which started/ended here before
+ if(style != vcTransport)
+ {
+ list<VPortConnection *>::iterator i;
+ stack<VPortConnection *> todo;
+
+ // reestablish transport connections which were ending at source...
+ for(i = source->incoming.begin(); i != source->incoming.end(); i++)
+ if((*i)->style != vcTransport) todo.push(*i);
+
+ // ... and starting at dest
+ for(i = dest->outgoing.begin(); i != dest->outgoing.end(); i++)
+ if((*i)->style != vcTransport) todo.push(*i);
+
+ // we need to do this with the stack as makeTransport can affect the
+ // incoming/outgoing lists by adding new vcTransport connections
+ while(!todo.empty())
+ {
+ todo.top()->source->makeTransport(todo.top());
+ todo.pop();
+ }
+ }
+
+#ifdef VPORT_DEBUG
+ cout << "~VPortConnection done" << endl;
+#endif
+}
+
+/*---------------------- virtual port implementation ----------------------*/
+
+VPort::VPort(Port *port) :port(port)
+{
+#ifdef VPORT_DEBUG
+ cout << "VPort: " << name() << endl;
+#endif
+}
+
+VPort::~VPort()
+{
+#ifdef VPORT_DEBUG
+ cout << "~VPort: " << name() << endl;
+#endif
+ while(!incoming.empty()) delete *incoming.begin();
+ while(!outgoing.empty()) delete *outgoing.begin();
+#ifdef VPORT_DEBUG
+ cout << "~VPort done" << endl;
+#endif
+}
+
+bool VPort::makeVirtualizeParams(VPort *forward, VPort*& source, VPort*& dest,
+ VPortConnection::Style& style)
+{
+ source = dest = 0;
+ // masquerading
+ if((port->flags() & streamIn) && (forward->port->flags() & streamIn))
+ {
+ // input: data flow direction is from us to the "forward" port
+ // XXX?
+ source = this;
+ dest = forward;
+ style = VPortConnection::vcMasquerade;
+ }
+ else if((port->flags() & streamOut) && (forward->port->flags() & streamOut))
+ {
+ // output: data flow direction is from the "forward" port to us
+ // XXX?
+ source = forward;
+ dest = this;
+ style = VPortConnection::vcMasquerade;
+ }
+ // forwarding
+ else if((port->flags() & streamIn) && (forward->port->flags() & streamOut))
+ {
+ source = this;
+ dest = forward;
+ style = VPortConnection::vcForward;
+ }
+ else if((port->flags() & streamOut) && (forward->port->flags() & streamIn))
+ {
+ source = forward;
+ dest = this;
+ style = VPortConnection::vcForward;
+ }
+ return source != 0;
+}
+
+/**
+ * a->virtualize(b) means, that the functionality that port a should provide
+ * (e.g. produce or consume data) is really provided by port b
+ */
+void VPort::virtualize(VPort *forward)
+{
+ VPort *source, *dest;
+ VPortConnection::Style style;
+
+ if(makeVirtualizeParams(forward,source,dest,style))
+ {
+#ifdef VPORT_DEBUG
+ cout << "virtualize ... source (producer) is " << source->name() <<
+ " dest (consumer) is " << dest->name() << endl;
+#endif
+ new VPortConnection(source,dest,style);
+ }
+}
+
+void VPort::devirtualize(VPort *forward)
+{
+ VPort *source, *dest;
+ VPortConnection::Style style;
+
+ // XXX?
+ if(makeVirtualizeParams(forward,source,dest,style))
+ {
+ list<VPortConnection *>::iterator i;
+ for(i = source->outgoing.begin(); i != source->outgoing.end(); i++)
+ {
+ if((*i)->source == source && (*i)->dest == dest
+ && (*i)->style == style)
+ {
+ delete (*i);
+ return;
+ }
+ }
+ }
+}
+
+void VPort::setFloatValue(float value)
+{
+ if(outgoing.empty())
+ {
+ AudioPort *aport = port->audioPort();
+ assert(aport);
+ aport->setFloatValue(value);
+ }
+ else
+ {
+ list<VPortConnection *>::iterator i;
+ for(i=outgoing.begin();i != outgoing.end(); i++)
+ {
+ VPortConnection *conn = *i;
+ assert(conn->style == VPortConnection::vcMasquerade);
+
+ conn->dest->setFloatValue(value);
+ }
+ }
+}
+
+void VPort::connect(VPort *dest)
+{
+ VPortConnection *conn;
+ if(port->flags() & streamOut)
+ {
+ conn = new VPortConnection(this,dest,VPortConnection::vcConnect);
+ }
+ else
+ {
+ conn = new VPortConnection(dest,this,VPortConnection::vcConnect);
+ }
+}
+
+void VPort::disconnect(VPort *dest)
+{
+ if(port->flags() & streamOut)
+ {
+ list<VPortConnection *>::iterator ci = outgoing.begin();
+ while(ci != outgoing.end())
+ {
+ assert((*ci)->source == this);
+ if((*ci)->dest == dest && (*ci)->style==VPortConnection::vcConnect)
+ {
+ delete (*ci); // will remove itself from the outgoing list
+ return;
+ }
+ ci++;
+ }
+ }
+ else
+ {
+ if(dest->port->flags() & streamOut)
+ {
+ dest->disconnect(this);
+ return;
+ }
+ }
+}
+
+void VPort::expandHelper(VPortConnection *conn, int state, VPort *current,
+ VPort *source, VPort *dest, bool remove)
+{
+ list<VPortConnection *>::iterator ci;
+
+#ifdef VPORT_DEBUG
+ cout << "expandhelper state " << state << " name " << current->name() << endl;
+#endif
+
+ if(state == 1) /* state 1: scan backward for output ports */
+ {
+ if(current->incoming.empty())
+ {
+ if(current->port->flags() & streamOut)
+ expandHelper(conn,2,current,current,dest,remove);
+ }
+ for(ci = current->incoming.begin(); ci != current->incoming.end();ci++)
+ {
+ assert((*ci)->style != VPortConnection::vcTransport);
+ expandHelper(conn,1,(*ci)->source,source,dest,remove);
+ }
+ }
+ else if(state == 2) /* state 2: output port expansion */
+ {
+ assert(current->port->flags() & streamOut);
+
+ for(ci = current->outgoing.begin(); ci != current->outgoing.end();ci++)
+ {
+ /* xconn=0 ensures that only paths are counted which contain conn */
+ VPortConnection *xconn = conn;
+ if(*ci == conn) xconn = 0;
+
+ if((*ci)->style == VPortConnection::vcMasquerade)
+ {
+ expandHelper(xconn,2,(*ci)->dest,source,dest,remove);
+ }
+ else if((*ci)->style == VPortConnection::vcConnect)
+ {
+ expandHelper(xconn,3,(*ci)->dest,source,(*ci)->dest,remove);
+ }
+ }
+ }
+ else if(state == 3) /* state 3: input port expansion */
+ {
+ assert(current->port->flags() & streamIn);
+
+ for(ci = current->outgoing.begin(); ci != current->outgoing.end();ci++)
+ {
+ /* xconn=0 ensures that only paths are counted which contain conn */
+ VPortConnection *xconn = conn;
+ if(*ci == conn) xconn = 0;
+
+ if((*ci)->style == VPortConnection::vcMasquerade)
+ {
+ // XXX ?
+ expandHelper(xconn,3,(*ci)->dest,source,(*ci)->dest,remove);
+ }
+ else if((*ci)->style == VPortConnection::vcForward)
+ {
+ expandHelper(xconn,2,(*ci)->dest,source,dest,remove);
+ }
+ }
+
+ if(current->outgoing.empty() && conn == 0)
+ {
+ if(remove)
+ {
+ // delete exactly one transport connection
+
+ bool removeOk = false;
+ ci = current->incoming.begin();
+ while(ci != current->incoming.end() && !removeOk)
+ {
+ if((*ci)->source == source && (*ci)->dest == dest
+ && (*ci)->style == VPortConnection::vcTransport)
+ {
+ delete (*ci);
+ removeOk = true;
+ }
+ else ci++;
+ }
+ assert(removeOk);
+ }
+ else
+ {
+ new VPortConnection(source,dest,VPortConnection::vcTransport);
+ }
+ }
+ }
+}
+
+void VPort::makeTransport(VPortConnection *conn)
+{
+ expandHelper(conn,1,this,0,0,false);
+}
+
+void VPort::removeTransport(VPortConnection *conn)
+{
+ expandHelper(conn,1,this,0,0,true);
+}
+
+const char *VPort::name()
+{
+ if(_name.empty())
+ {
+ _name = port->parent->object()->_interfaceName() + "." +
+ port->name();
+ }
+ return _name.c_str();
+}