diff options
Diffstat (limited to 'flow/virtualports.cc')
-rw-r--r-- | flow/virtualports.cc | 491 |
1 files changed, 0 insertions, 491 deletions
diff --git a/flow/virtualports.cc b/flow/virtualports.cc deleted file mode 100644 index b733256..0000000 --- a/flow/virtualports.cc +++ /dev/null @@ -1,491 +0,0 @@ - /* - - Copyright (C) 2000 Stefan Westerfeld - stefan@space.twc.de - - 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(); -} |