1 /*=========================================================================
3 Module: $RCSfile: bbtkConnection.cxx,v $
5 Date: $Date: 2008/12/08 12:54:19 $
6 Version: $Revision: 1.16 $
7 =========================================================================*/
9 /* ---------------------------------------------------------------------
11 * Copyright (c) CREATIS-LRMN (Centre de Recherche en Imagerie Medicale)
12 * Authors : Eduardo Davila, Laurent Guigues, Jean-Pierre Roux
14 * This software is governed by the CeCILL-B license under French law and
15 * abiding by the rules of distribution of free software. You can use,
16 * modify and/ or redistribute the software under the terms of the CeCILL-B
17 * license as circulated by CEA, CNRS and INRIA at the following URL
18 * http://www.cecill.info/licences/Licence_CeCILL-B_V1-en.html
19 * or in the file LICENSE.txt.
21 * As a counterpart to the access to the source code and rights to copy,
22 * modify and redistribute granted by the license, users are provided only
23 * with a limited warranty and the software's author, the holder of the
24 * economic rights, and the successive licensors have only limited
27 * The fact that you are presently reading this means that you have had
28 * knowledge of the CeCILL-B license and that you accept its terms.
29 * ------------------------------------------------------------------------ */
33 *\brief Class bbtk::Connection
36 #include "bbtkConnection.h"
37 #include "bbtkFactory.h"
38 #include "bbtkBlackBox.h"
39 #include "bbtkMessageManager.h"
40 #include "bbtkBlackBoxOutputConnector.h"
44 const std::string IOSTATUS_STRING[3] =
45 {"Up-to-date","Modified","Out-of-date"};
47 const std::string& GetIOStatusString(IOStatus s)
48 { return IOSTATUS_STRING[s]; }
50 //==================================================================
51 Connection::Pointer Connection::New(BlackBox::Pointer from,
52 const std::string& output,
54 const std::string& input ,
55 const Factory::Pointer f )
57 bbtkDebugMessage("object",1,"##> Connection::Connection(\""
58 <<from->bbGetName()<<"\",\""<<output<<"\",\""
59 <<to->bbGetName()<<"\",\""<<input<<"\")"
61 Connection::Pointer p =
62 MakePointer(new Connection(from,output,to,input,f));
63 bbtkDebugMessage("object",1,"<## Connection::Connection(\""
64 <<from->bbGetName()<<"\",\""<<output<<"\",\""
65 <<to->bbGetName()<<"\",\""<<input<<"\")"
69 //==================================================================
71 //==================================================================
72 /// Ctor with the black box from and to and their input and output.
73 /// Check the input and output compatibility
74 Connection::Connection(BlackBox::Pointer from, const std::string& output,
75 BlackBox::Pointer to, const std::string& input ,
76 const Factory::Pointer f )
82 bbtkDebugMessage("object",2,"==> Connection::Connection(\""
83 <<from->bbGetName()<<"\",\""<<output<<"\",\""
84 <<to->bbGetName()<<"\",\""<<input<<"\")"
87 bbtkDebugMessage("connection",1,"==> Connection::Connection(\""
88 <<from->bbGetFullName()<<"\",\""<<output<<"\",\""
89 <<to->bbGetFullName()<<"\",\""<<input<<"\")"
94 if (! from->bbHasOutput(output) )
96 bbtkError("The box \""<<from->bbGetTypeName()<<
97 "\" has no output \""<<output<<"\"");
99 if (! to->bbHasInput(input) )
101 bbtkError("The box \""<<to->bbGetTypeName()<<
102 "\" has no input \""<<input<<"\"");
105 if (to->bbGetInputConnectorMap().find(input)->second->IsConnected())
107 bbtkError("The input \""<<input<<"\" of the box \""<<to->bbGetName()
108 <<"\" is already connected");
111 // std::string t1 ( from->bbGetOutputType(output).name() );
112 // std::string t2 ( to->bbGetInputType(input).name() );
114 if ( from->bbGetOutputType(output) !=
115 to->bbGetInputType(input) )
117 if ( from->bbGetOutputType(output) == typeid(Data) )
119 bbtkWarning("Connection '"
121 <<"' : '"<<from->bbGetName()<<"."<<output
123 <<HumanTypeName<Data>()
124 <<"> : type compatibility with '"
125 <<to->bbGetName()<<"."<<input
126 <<"' will be resolved at run time"
130 else if ( to->bbGetInputType(input) == typeid(Data) )
132 bbtkDebugMessage("Kernel",8," -> '"<<input<<"' type is "
133 <<TypeName<Data>()<<" : can receive any data"
139 // std::cout << "Adaptive connection "<<std::endl;
141 name = from->bbGetName() + "." + output + "-"
142 + to->bbGetName() + "." + input;
143 mAdaptor = mFactory.lock()
144 ->NewAdaptor(from->bbGetOutputType(output),
145 to->bbGetInputType(input),
149 bbtkError("did not find any <"
150 <<TypeName(from->bbGetOutputType(output))
152 <<TypeName(to->bbGetInputType(input))
160 mOriginalFrom = from;
163 mInput = mOriginalInput = input;
164 mOutput = mOriginalOutput = output;
166 // Lock this pointer !!!
167 //Pointer p = MakePointer(this,true);
168 from->bbConnectOutput(output,this);
169 to->bbConnectInput(input,this);
171 from->bbGetOutputConnector(output).AddChangeObserver(boost::bind(&bbtk::Connection::OnOutputChange,this, _1, _2, _3));
174 bbtkDebugMessage("connection",1,"<== Connection::Connection(\""
175 <<from->bbGetFullName()<<"\",\""<<output<<"\",\""
176 <<to->bbGetFullName()<<"\",\""<<input<<"\")"
179 bbtkDebugMessage("object",2,"==> Connection::Connection(\""
180 <<from->bbGetName()<<"\",\""<<output<<"\",\""
181 <<to->bbGetName()<<"\",\""<<input<<"\")"
184 //==================================================================
186 //==================================================================
187 Connection::Pointer Connection::New(BlackBox::Pointer from,
188 const std::string& output,
189 BlackBox::Pointer to,
190 const std::string& input )
192 bbtkDebugMessage("object",1,"##> Connection::Connection(\""
193 <<from->bbGetName()<<"\",\""<<output<<"\",\""
194 <<to->bbGetName()<<"\",\""<<input<<"\")"
196 Connection::Pointer p =
197 MakePointer(new Connection(from,output,to,input));
198 bbtkDebugMessage("object",1,"<## Connection::Connection(\""
199 <<from->bbGetName()<<"\",\""<<output<<"\",\""
200 <<to->bbGetName()<<"\",\""<<input<<"\")"
204 //==================================================================
206 //==================================================================
207 /// Ctor with the black box from and to and their input and output.
208 /// Check the input and output compatibility
209 Connection::Connection(BlackBox::Pointer from, const std::string& output,
210 BlackBox::Pointer to, const std::string& input )
216 bbtkDebugMessage("object",2,"==> Connection::Connection(\""
217 <<from->bbGetName()<<"\",\""<<output<<"\",\""
218 <<to->bbGetName()<<"\",\""<<input<<"\")"
221 bbtkDebugMessage("connection",1,"==> Connection::Connection(\""
222 <<from->bbGetFullName()<<"\",\""<<output<<"\",\""
223 <<to->bbGetFullName()<<"\",\""<<input<<"\")"
228 if (! from->bbHasOutput(output) )
230 bbtkError("The box \""<<from->bbGetTypeName()<<
231 "\" has no output \""<<output<<"\"");
233 if (! to->bbHasInput(input) )
235 bbtkError("The box \""<<to->bbGetTypeName()<<
236 "\" has no input \""<<input<<"\"");
239 if (to->bbGetInputConnectorMap().find(input)->second->IsConnected())
241 bbtkError("The input \""<<input<<"\" of the box \""<<to->bbGetName()
242 <<"\" is already connected");
245 // std::string t1 ( from->bbGetOutputType(output).name() );
246 // std::string t2 ( to->bbGetInputType(input).name() );
248 if ( from->bbGetOutputType(output) !=
249 to->bbGetInputType(input) )
251 if ( from->bbGetOutputType(output) == typeid(Data) )
253 bbtkWarning("Connection '"
255 <<"' : '"<<from->bbGetName()<<"."<<output
257 <<HumanTypeName<Data>()
258 <<"> : type compatibility with '"
259 <<to->bbGetName()<<"."<<input
260 <<"' will be resolved at run time"
264 else if ( to->bbGetInputType(input) == typeid(Data) )
266 bbtkDebugMessage("Kernel",8," -> '"<<input<<"' type is "
267 <<TypeName<Data>()<<" : can receive any data"
273 bbtkError("Connection created between different types without Factory provided");
279 mOriginalFrom = from;
282 mInput = mOriginalInput = input;
283 mOutput = mOriginalOutput = output;
285 // Lock this pointer !!!
286 //Pointer p = MakePointer(this,true);
287 from->bbConnectOutput(output,this);
288 to->bbConnectInput(input,this);
290 from->bbGetOutputConnector(output).AddChangeObserver(boost::bind(&bbtk::Connection::OnOutputChange,this, _1, _2, _3));
292 bbtkDebugMessage("connection",1,"<== Connection::Connection(\""
293 <<from->bbGetFullName()<<"\",\""<<output<<"\",\""
294 <<to->bbGetFullName()<<"\",\""<<input<<"\")"
297 bbtkDebugMessage("object",2,"==> Connection::Connection(\""
298 <<from->bbGetName()<<"\",\""<<output<<"\",\""
299 <<to->bbGetName()<<"\",\""<<input<<"\")"
302 //==================================================================
304 //==================================================================
306 Connection::~Connection()
308 bbtkDebugMessage("object",2,
309 "==> Connection::~Connection() ["
310 <<GetFullName()<<"]"<<std::endl);
312 if (mAdaptor) mAdaptor.reset();
315 mFrom->bbDisconnectOutput(mOutput,this);
316 // GetThisPointer<Connection>());
321 bbtkInternalError("Connection::~Connection() : invalid initial box pointer");
325 mTo->bbDisconnectInput(mInput,this);// GetThisPointer<Connection>());
330 bbtkInternalError("Connection::~Connection() : invalid final box pointer");
334 bbtkDebugMessage("object",2,
335 "<== Connection::~Connection() ["
336 <<GetFullName()<<"]"<<std::endl);
338 //==================================================================
340 //==================================================================
342 void Connection::BackwardUpdate()
344 bbtkDebugMessage("process",5,
345 "===> Connection::BackwardUpdate() ["
346 <<GetFullName()<<"]"<<std::endl);
348 mFrom->bbBackwardUpdate(GetThisPointer<Connection>());
353 IOStatus s = MODIFIED;
354 if ( mFrom->bbGetOutputConnector(mOutput).GetStatus() == OUTOFDATE)
356 mTo->bbGetInputConnector(mInput).SetStatus(s);
358 bbtkDebugMessage("process",5,
359 "<=== Connection::BackwardUpdate() ["
360 <<GetFullName()<<"]"<<std::endl);
363 //==================================================================
366 //==================================================================
368 void Connection::ForwardUpdate()
370 bbtkDebugMessageInc("process",2,
371 "Connection::ForwardUpdate() ["
372 <<GetFullName()<<"]"<<std::endl);
377 mTo->bbForwardUpdate(this);
379 bbtkDebugDecTab("process",2);
381 //==================================================================
384 //==================================================================
385 /// Transfers the data from the source output to the target input
386 /// doing necessary conversions (adaptation or pointer cast)
387 void Connection::TransferData()
389 bbtkDebugMessageInc("data",3,
390 "Connection::TransferData() ["
391 <<GetFullName()<<"]"<<std::endl);
394 // If an adaptor was created we need to adapt the data
397 mAdaptor->bbSetInput("In",mFrom->bbGetOutput(mOutput),false);
398 mAdaptor->bbExecute();
399 // LG : Connection Update does not set mTo as modified
400 mTo->bbSetInput(mInput, mAdaptor->bbGetOutput("Out"),false);
403 // If no adaptor but source type is an any and target is not an any
404 else if ( mFromAny && (! mToAny) )
406 bbtkDebugMessage("data",3,
407 " * Source type is an "
408 <<HumanTypeName<Data>()
409 <<" which contains a <"
410 <<HumanTypeName(mFrom->bbGetOutput(mOutput).type())
412 bbtkDebugMessage("data",3,
413 " * Target type is <"
414 <<HumanTypeName(mTo->bbGetInputType(mInput))
417 // 1) Test strict type matching between any content and target
418 if (mFrom->bbGetOutput(mOutput)
419 .contains( mTo->bbGetInputType(mInput) ) )
421 bbtkDebugMessage("data",3,
422 " -> Equal types : transfer ok"<<std::endl);
423 mTo->bbSetInput( mInput,
424 mFrom->bbGetOutput(mOutput),
429 // 2) Look for an adaptor
430 bbtk::BlackBox::Pointer adaptor;
433 adaptor = mFactory.lock()
434 ->NewAdaptor(mFrom->bbGetOutput(mOutput).type(),
435 mTo->bbGetInputType(mInput),
443 bbtkDebugMessage("data",3," -> Adaptor found : using it"
445 adaptor->bbSetInput("In",mFrom->bbGetOutput(mOutput),false);
446 adaptor->bbExecute();
447 // LG : Connection Update does not set mTo as modified
448 mTo->bbSetInput(mInput, adaptor->bbGetOutput("Out"),false);
449 // adaptor->bbDelete();
451 // 3) If no adaptor found but the any content is a pointer
452 // and target type is also a pointer : we try run-time cast
453 else if ( (mFrom->bbGetOutput(mOutput).contains_pointer()) &&
454 (mTo->bbGetDescriptor()->GetInputDescriptor(mInput)
457 bbtkDebugMessage("data",3,
458 " -> No adaptor found but source and target types are both pointers : trying up or down cast"<<std::endl);
461 mFrom->bbGetOutput(mOutput)
462 .get_pointer_to(mTo->bbGetInput(mInput).pointed_type());
465 bbtkError("Connection '"
468 <<HumanTypeName(mFrom->bbGetOutput(mOutput).type())
470 <<HumanTypeName(mTo->bbGetInputType(mInput))
471 <<"> : no adaptor available and run-time up and down cast failed");
473 mTo->bbBruteForceSetInputPointer(mInput, nptr, false);
475 // 4) Nothing worked : error
478 bbtkError("Connection '"<<GetFullName()<<"' "
479 <<"no adaptor found to convert <"
480 <<HumanTypeName(mFrom->bbGetOutput(mOutput).type())
482 <<HumanTypeName(mTo->bbGetInputType(mInput))<<">");
486 // EO : mFromAny && ! mToAny
487 // Default case : types are the same; we use simple get-set
490 // LG : Connection Update does not set mTo as modified
491 mTo->bbSetInput(mInput, mFrom->bbGetOutput(mOutput),false);
495 //==================================================================
498 //==================================================================
500 void Connection::SetModifiedStatus()
502 bbtkDebugMessage("modified",2,
503 "==> Connection::SetModifiedStatus() ["
504 <<GetFullName()<<"]"<<std::endl);
506 if (mAdaptor) mAdaptor->bbSetModifiedStatus();
508 mTo->bbSetModifiedStatus( mTo->bbGetInputConnectorMap().find(mInput)->second );
512 //==================================================================
514 //==================================================================
515 /// From.Output change propagation
516 void Connection::OnOutputChange(bbtk::BlackBox::Pointer, const std::string&,
519 bbtkDebugMessage("change",2,
520 "==> Connection::OnOutputChange("<<status<<") ["
521 <<GetFullName()<<"]"<<std::endl);
525 BlackBoxInputConnector* ac = mAdaptor->bbGetInputConnectorMap().find("In")->second;
526 mAdaptor->bbSetStatusAndPropagate(ac,status);
529 mTo->bbSetStatusAndPropagate( mTo->bbGetInputConnectorMap().find(mInput)->second, status);
533 //==================================================================
536 //==================================================================
537 std::string Connection::GetFullName() const {
540 std::string res = mFrom->bbGetName()+"."+mOutput+"--"
541 +mTo->bbGetName()+"."+mInput;
542 if ((!mOriginalFrom.expired()) && (!mOriginalTo.expired()) &&
543 ((mFrom!=mOriginalFrom.lock())||(mTo!=mOriginalTo.lock())))
545 res += "("+mOriginalFrom.lock()->bbGetName()
546 +"."+mOriginalOutput+"--"
547 + mOriginalTo.lock()->bbGetName()+"."+mOriginalInput+")";
551 return "***Invalid Connection***";
553 //==================================================================
555 //==================================================================
556 void Connection::Check() const
558 bbtkMessage("debug",1,"** Checking Connection "<<(void*)this<<" ["<<GetFullName()<<"]"
562 bbtkMessage("debug",2," - From = 0"<<std::endl);
566 bbtkMessage("debug",2," - From : "<<mFrom->bbGetFullName()<<std::endl);
567 if (!mFrom->bbHasOutput(mOutput))
569 bbtkError("** Checking Connection "<<(void*)this
570 <<" ["<<GetFullName()<<"] : "
571 << mFrom->bbGetFullName()<<" does not have output '"
574 bbtkMessage("debug",2," - From : Output '"<<mOutput<<"' exists"<<std::endl);
575 BlackBox::OutputConnectorMapType::const_iterator i
576 = mFrom->bbGetOutputConnectorMap().find(mOutput);
577 if (i== mFrom->bbGetOutputConnectorMap().end())
579 bbtkError("** Checking Connection "<<(void*)this
580 <<" ["<<GetFullName()<<"] : "
581 <<mFrom->bbGetFullName()<<" output '"
582 <<mOutput<<"' is not in OutputConnectorMap");
584 bbtkMessage("debug",2," - From : Output '"<<mOutput
585 <<"' is in OutputConnectorMap"<<std::endl);
587 std::vector< Connection* >::const_iterator j;
589 for (j = i->second->GetConnectionVector().begin();
590 j != i->second->GetConnectionVector().end();
593 if ((*j)==this) break;
596 j = find(i->second->GetConnectionVector().begin(),
597 i->second->GetConnectionVector().end(),
600 if (j==i->second->GetConnectionVector().end())
602 bbtkError("** Checking Connection "<<(void*)this
603 <<" ["<<GetFullName()<<"] : "
604 << "Connection ["<<GetFullName()<<"] : "
605 <<" OutputConnector '"
606 <<mOutput<<"' of "<<mFrom->bbGetFullName()
607 <<" does not point to this connection");
610 bbtkMessage("debug",2," - From : This connection is in OutputConnector connection vector"<<std::endl);
611 bbtkMessage("debug",2," * Box from : Check successfull"<<std::endl);
617 bbtkMessage("debug",2," - To = 0"<<std::endl);
621 bbtkMessage("debug",2," - To : "<<mTo->bbGetName()<<std::endl);
622 // std::cout << mTo << std::endl;
623 // std::cout << mTo->bbGetDescriptor() << std::endl;
624 // std::cout << mTo->bbGetDescriptor()->GetTypeName() << std::endl;
625 // mTo->bbGetFullName();
626 bbtkMessage("debug",2," - To : "<<mTo->bbGetFullName()<<std::endl);
627 if (!mTo->bbHasInput(mInput))
629 bbtkError("** Checking Connection "<<(void*)this
630 <<" ["<<GetFullName()<<"] : "
631 <<mTo->bbGetFullName()<<" does not have input '"
634 bbtkMessage("debug",2," - To : Input '"<<mInput<<"' exists"<<std::endl);
635 BlackBox::InputConnectorMapType::const_iterator i
636 = mTo->bbGetInputConnectorMap().find(mInput);
637 if (i== mTo->bbGetInputConnectorMap().end())
639 bbtkError("** Checking Connection "<<(void*)this
640 <<" ["<<GetFullName()<<"] : "
641 <<mTo->bbGetFullName()<<" input '"
642 <<mInput<<"' is not in InputConnectorMap");
644 bbtkMessage("debug",2," - To : Input '"<<mInput
645 <<"' is in InputConnectorMap"<<std::endl);
647 if (i->second->GetConnection()==0)
649 bbtkError("** Checking Connection "<<(void*)this
650 <<" ["<<GetFullName()<<"] : "
651 <<"Connection "<<GetFullName()<<" : "
652 <<" InputConnector '"
653 <<mInput<<"' of "<<mTo->bbGetFullName()
654 <<" does not point to this connection");
657 bbtkMessage("debug",2," - To : This connection is in InputConnector connection vector"<<std::endl);
658 bbtkMessage("debug",2," * Box to : Check successfull"<<std::endl);
662 //==================================================================
663 //==========================================================================
664 std::string Connection::GetObjectName() const
666 std::string s("Connection '");
671 //==========================================================================
673 //==========================================================================
674 std::string Connection::GetObjectInfo() const
679 //==========================================================================
681 //==========================================================================
682 size_t Connection::GetObjectSize() const
684 size_t s = Superclass::GetObjectSize();
685 s += Connection::GetObjectInternalSize();
688 //==========================================================================
689 //==========================================================================
690 size_t Connection::GetObjectInternalSize() const
692 size_t s = sizeof(Connection);
695 //==========================================================================
696 //==========================================================================
697 size_t Connection::GetObjectRecursiveSize() const
699 size_t s = Superclass::GetObjectRecursiveSize();
700 s += Connection::GetObjectInternalSize();
703 //==========================================================================