Orocos Real-Time Toolkit  2.6.0
FunctionGraphBuilder.cpp
00001 /***************************************************************************
00002   tag: Peter Soetens  Wed Jan 18 14:11:40 CET 2006  FunctionGraphBuilder.cxx
00003 
00004                         FunctionGraphBuilder.cxx -  description
00005                            -------------------
00006     begin                : Wed January 18 2006
00007     copyright            : (C) 2006 Peter Soetens
00008     email                : peter.soetens@mech.kuleuven.be
00009 
00010  ***************************************************************************
00011  *   This library is free software; you can redistribute it and/or         *
00012  *   modify it under the terms of the GNU General Public                   *
00013  *   License as published by the Free Software Foundation;                 *
00014  *   version 2 of the License.                                             *
00015  *                                                                         *
00016  *   As a special exception, you may use this file as part of a free       *
00017  *   software library without restriction.  Specifically, if other files   *
00018  *   instantiate templates or use macros or inline functions from this     *
00019  *   file, or you compile this file and link it with other files to        *
00020  *   produce an executable, this file does not by itself cause the         *
00021  *   resulting executable to be covered by the GNU General Public          *
00022  *   License.  This exception does not however invalidate any other        *
00023  *   reasons why the executable file might be covered by the GNU General   *
00024  *   Public License.                                                       *
00025  *                                                                         *
00026  *   This library is distributed in the hope that it will be useful,       *
00027  *   but WITHOUT ANY WARRANTY; without even the implied warranty of        *
00028  *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU     *
00029  *   Lesser General Public License for more details.                       *
00030  *                                                                         *
00031  *   You should have received a copy of the GNU General Public             *
00032  *   License along with this library; if not, write to the Free Software   *
00033  *   Foundation, Inc., 59 Temple Place,                                    *
00034  *   Suite 330, Boston, MA  02111-1307  USA                                *
00035  *                                                                         *
00036  ***************************************************************************/
00037 
00038 
00039 
00040 #include "FunctionGraphBuilder.hpp"
00041 #include "CommandComposite.hpp"
00042 #include "../base/AttributeBase.hpp"
00043 #include "../internal/DataSource.hpp"
00044 //#include "parse_exception.hpp"
00045 #include "../FactoryExceptions.hpp"
00046 #include "GraphCopier.hpp"
00047 
00048 #include "CommandNOP.hpp"
00049 #include "ConditionFalse.hpp"
00050 #include "ConditionTrue.hpp"
00051 #include "ConditionTrue.hpp"
00052 #include <boost/graph/copy.hpp>
00053 #include <utility>
00054 
00055 #include <iostream>
00056 
00057 namespace RTT {
00058     using namespace detail;
00059     using namespace boost;
00060     using namespace std;
00061 
00062 
00063 
00064 
00065     FunctionGraphBuilder::FunctionGraphBuilder()
00066         : graph( 0 )
00067     {
00068     }
00069 
00070     FunctionGraphBuilder::~FunctionGraphBuilder()
00071     {
00072     }
00073 
00074     FunctionGraphPtr FunctionGraphBuilder::startFunction(const std::string& fname)
00075     {
00076         // next node should be 'empty'/ not used here.
00077         // a function is to be constructed, it will unload when it stops/errors
00078         func.reset( new FunctionGraph( fname, true ) );
00079         graph   = &func->getGraph();
00080         build   = func->startNode();
00081         next    = add_vertex( *graph );
00082         put( vertex_exec, *graph, next, VertexNode::normal_node );
00083         return func;
00084     }
00085 
00086     void FunctionGraphBuilder::returnFunction( ConditionInterface* cond, int line )
00087     {
00088         // connect the build node to the exitNode under a condition,
00089         // for example, the build implicit term condition.
00090         add_edge(build, func->exitNode(), EdgeCondition(cond), *graph);
00091     }
00092 
00093     FunctionGraphPtr FunctionGraphBuilder::getFunction()
00094     {
00095         return func;
00096     }
00097 
00098     FunctionGraphPtr FunctionGraphBuilder::endFunction( int endline )
00099     {
00100         // the map contains _references_ to all vertex_command properties
00101         boost::property_map<Graph, vertex_command_t>::type
00102             cmap = get(vertex_command, *graph);
00103         cmap[func->exitNode()].setLineNumber( endline );
00104 
00105         // A return statement is obligatory, so the returnFunction has
00106         // already connected the build node to exitNode of the function.
00107         // the end of the statement will proceedToNext
00108         // remove the empty next nodes.
00109         remove_vertex( build, *graph );
00110         remove_vertex( next, *graph );
00111 
00112         func->finish();
00113         func->reset();
00114 
00115         FunctionGraphPtr tfunc = func;
00116         func.reset();
00117         return tfunc;
00118     }
00119 
00120     FunctionGraphBuilder::CommandNode FunctionGraphBuilder::addCommand( ConditionInterface* cond,  ActionInterface* com )
00121     {
00122         add_edge(build, next, EdgeCondition(cond), *graph);
00123         build = next;
00124         setCommand(com);
00125         next    = add_vertex( *graph );
00126         put( vertex_exec, *graph, next, VertexNode::normal_node );
00127         return next;
00128     }
00129 
00130     void FunctionGraphBuilder::addConditionEdge( ConditionInterface* cond, CommandNode vert )
00131     {
00132         add_edge(build, vert, EdgeCondition(cond), *graph);
00133     }
00134 
00135     void FunctionGraphBuilder::closeConditionEdge( CommandNode vert, ConditionInterface* cond )
00136     {
00137         add_edge(vert, build, EdgeCondition(cond), *graph);
00138     }
00139 
00140     FunctionGraphBuilder::CommandNode FunctionGraphBuilder::moveTo( CommandNode _build, CommandNode _next, int linenumber )
00141     {
00142         this->setLineNumber( linenumber );
00143         CommandNode old = build;
00144         build = _build;
00145         next    = _next;
00146         return old;
00147     }
00148 
00149     ActionInterface* FunctionGraphBuilder::getCommand( CommandNode cn )
00150     {
00151         // the map contains _references_ to all vertex_command properties
00152         boost::property_map<Graph, vertex_command_t>::type
00153             cmap = get(vertex_command, *graph);
00154         // access the one of build
00155         return cmap[cn].getCommand();
00156     }
00157 
00158     void FunctionGraphBuilder::setCommand( ActionInterface* comm )
00159     {
00160         this->setCommand(build, comm);
00161     }
00162 
00163     void FunctionGraphBuilder::setCommand(CommandNode cn, ActionInterface* comm )
00164     {
00165         // the map contains _references_ to all vertex_command properties
00166         boost::property_map<Graph, vertex_command_t>::type
00167             cmap = get(vertex_command, *graph);
00168         // access the one of build
00169         delete cmap[cn].setCommand( comm );
00170     }
00171 
00172     void FunctionGraphBuilder::setName(const std::string& pname) {
00173         func->setName(pname);
00174     }
00175 
00176     FunctionGraphBuilder::CommandNode FunctionGraphBuilder::proceedToNext( ConditionInterface* cond, int this_line )
00177     {
00178         add_edge(build, next, EdgeCondition(cond), *graph);
00179         return proceedToNext( this_line );
00180     }
00181 
00182     FunctionGraphBuilder::CommandNode FunctionGraphBuilder::proceedToNext( int this_line )
00183     {
00184         if ( this_line )
00185             this->setLineNumber( this_line );
00186         build = next;
00187         next    = add_vertex( *graph );
00188         put(vertex_exec, *graph, next, VertexNode::normal_node );
00189         return build;
00190     }
00191 
00192     void FunctionGraphBuilder::connectToNext( CommandNode v, ConditionInterface* cond )
00193     {
00194         add_edge( v, next, EdgeCondition(cond), *graph);
00195     }
00196 
00197     FunctionGraphBuilder::CommandNode FunctionGraphBuilder::buildNode() const
00198     {
00199         return build;
00200     }
00201 
00202     size_t FunctionGraphBuilder::buildEdges() const
00203     {
00204         return out_degree( build, *graph );
00205     }
00206 
00207     FunctionGraphBuilder::CommandNode FunctionGraphBuilder::nextNode() const
00208     {
00209         return next;
00210     }
00211 
00212     void FunctionGraphBuilder::setLineNumber( int line )
00213     {
00214         boost::property_map<Graph, vertex_command_t>::type
00215             cmap = get(vertex_command, *graph);
00216         cmap[build].setLineNumber( line );
00217     }
00218 
00219 
00220     template< class _Map >
00221     struct finder {
00222         typedef FunctionGraphBuilder::Vertex first_argument_type ;
00223         typedef std::pair<_Map, int> second_argument_type ;
00224         typedef bool result_type;
00225         bool operator()(const FunctionGraphBuilder::Vertex& v,
00226                         const std::pair<_Map, int>& to_find  ) const
00227             {
00228                 // return : Is this the node with the given property ?
00229                 return to_find.first[v] == to_find.second;
00230             }
00231     };
00232 
00233     FunctionGraphBuilder::CommandNode FunctionGraphBuilder::appendFunction( ConditionInterface* cond, FunctionGraphPtr fn,
00234                                                             std::vector<DataSourceBase::shared_ptr> fnargs )
00235     {
00239         // Do not move build inhere !
00240         // copy FunctionGraph's graph into the current *graph, add an edge with
00241         // the condition. Connect the exitNode() of the function with
00242         // the next node.
00243 
00244         // The reason to reindex the vertexes of source graph is
00245         // to silence valgrinds warnings about uninitialised values
00246         // in the following piece of code. Technically, copy_graph
00247         // does not need it.
00248         boost::property_map<Graph, vertex_index_t>::type indexmap =
00249             get( vertex_index, *graph );
00250         boost::graph_traits<Graph>::vertex_iterator v, vend;
00251         int index = 0;
00252         for ( tie( v, vend ) = vertices( *graph ); v != vend; ++v )
00253             indexmap[*v] = index++;
00254 
00255         // check nb of arguments:
00256         std::vector<AttributeBase*> origlist = fn->getArguments();
00257         if ( fnargs.size() != origlist.size() ) {
00258 #ifndef ORO_EMBEDDED
00259             throw wrong_number_of_args_exception( origlist.size(), fnargs.size() );
00260 #else
00261             return buildNode();
00262 #endif
00263         }
00264 
00265         // make a deep copy of the function :
00266         std::map<const DataSourceBase*, DataSourceBase*> replacementdss;
00267         std::vector<AttributeBase*> newlist;
00268         for (unsigned int i=0; i < origlist.size(); ++i)
00269             newlist.push_back( origlist[i]->copy( replacementdss, false ) ); // args are not instantiated.
00270         // newlist contains the DS which need initialisations :
00271 
00272         // create commands that init all the args :
00273         CommandComposite* icom=  new CommandComposite();
00274         std::vector<DataSourceBase::shared_ptr>::const_iterator dit = fnargs.begin();
00275         std::vector<AttributeBase*>::const_iterator tit =  newlist.begin();
00276 #ifndef ORO_EMBEDDED
00277         try {
00278             for (; dit != fnargs.end(); ++dit, ++tit)
00279                 icom->add( (*tit)->getDataSource()->updateAction( dit->get() ) );
00280         }
00281         catch( const bad_assignment& ) {
00282             // cleanup allocated memory
00283             for (unsigned int i=0; i < newlist.size(); ++i)
00284                 delete newlist[i];
00285             delete icom;
00286             int parnb = (dit - fnargs.begin());
00287             throw wrong_types_of_args_exception(parnb, (*tit)->getDataSource()->getType() ,(*dit)->getType() );
00288         }
00289 #else
00290         for (; dit != fnargs.end(); ++dit, ++tit) {
00291             ActionInterface* ret = (*tit)->getDataSource()->updateAction( dit->get() );
00292             if (ret)
00293                 icom->add( ret );
00294             else {
00295                 // cleanup allocated memory
00296                 for (unsigned int i=0; i < newlist.size(); ++i)
00297                     delete newlist[i];
00298                 delete icom;
00299                 return buildNode();
00300             }
00301         }
00302 #endif
00303         // set the init command on the build node
00304         //assert( build not used by other than NOP )
00305         assert( dynamic_cast<CommandNOP*>( this->getCommand(build) ));
00306         this->setCommand( icom );
00307 
00308         boost::copy_graph( fn->getGraph(), *graph,
00309                            boost::vertex_copy( GraphVertexCopier( fn->getGraph(), *graph, replacementdss ) ).
00310                            edge_copy( GraphEdgeCopier( fn->getGraph(), *graph, replacementdss ) ) );
00311 
00312         // cleanup newlist, the (var)DS's are stored in the assignCommand
00313         for (unsigned int i=0; i < newlist.size(); ++i)
00314             delete newlist[i];
00315 
00316         // the subgraph has been copied but is now 'floating' in the current graph.
00317         // we search func start and exit points and connect them to
00318         // the current graph.
00319 
00320         // domi: it would be cleaner if a function would keep a
00321         // reference to its enter and exit point, and we would use the
00322         // copy_graph orig_to_copy map to find the corresponding nodes
00323         // in the copy...
00324         graph_traits<Graph>::vertex_iterator v1,v2, vc;
00325         tie(v1,v2) = vertices(*graph);
00326         boost::property_map<Graph, vertex_exec_t>::type
00327             vmap = get(vertex_exec, *graph);
00328 
00329 //         // test not initialised value.
00330 //         vc = v1;
00331 //         while (vc != v2) {
00332 //             std::cerr << int( vmap[*vc] )<<std::endl;
00333 //             ++vc;
00334 //         }
00335 //         std::cerr<<std::endl;
00336 //         // test foo :
00337 //         tie(v1,v2) = vertices(fn->getGraph());
00338 //             vmap = get(vertex_exec, fn->getGraph());
00339 //         vc = v1;
00340 //         while (vc != v2) {
00341 //             std::cerr << int( vmap[*vc] )<<std::endl;
00342 //             ++vc;
00343 //         }
00344 
00345 
00346         Vertex funcStart=*
00347                          find_if(v1, v2,
00348                                  bind2nd( finder<boost::property_map<Graph, vertex_exec_t>::type>() ,
00349                                           std::make_pair( vmap, int(VertexNode::func_start_node)) ) );
00350         Vertex funcExit=*
00351                         find_if(v1, v2,
00352                                 bind2nd( finder<boost::property_map<Graph, vertex_exec_t>::type>() ,
00353                                          std::make_pair( vmap, int(VertexNode::func_exit_node)) ) );
00354 
00355         // reset their special meanings.
00356         vmap[funcStart] = VertexNode::normal_node;
00357         vmap[funcExit]  = VertexNode::normal_node;
00358         // connect the graph.
00359         addConditionEdge( cond, funcStart );
00360         connectToNext( funcExit, new ConditionTrue );
00361 
00362         return funcExit;
00363 //         // try to solve call in function bug with arguments
00364 //         boost::property_map<Graph, vertex_command_t>::type
00365 //             cmap = get(vertex_command, *graph);
00366 //         return proceedToNext( cmap[funcExit].getLineNumber() );
00367     }
00368 
00369     FunctionGraphBuilder::CommandNode FunctionGraphBuilder::setFunction( FunctionGraphPtr fn,
00370                                                          std::vector<DataSourceBase::shared_ptr> fnargs )
00371     {
00380         return appendFunction( new ConditionTrue, fn, fnargs) ;
00381     }
00382 
00383     void FunctionGraphBuilder::startIfStatement( ConditionInterface* cond, int linenumber )
00384     {
00385         // push all relevant nodes on the branch_stack.
00386         // endIf and endElse will pop them
00387 
00388         // * next will become the first node of the succeeding if statement
00389         // * after_else_node will become the node after the if block is finished
00390         // and the else block is finished.
00391         // * else_node is the first node of a failing if statement (this may be _any_statement, thus also an if)
00392         CommandNode else_node = add_vertex( *graph );
00393         put(vertex_exec, *graph, else_node, VertexNode::normal_node );
00394         branch_stack.push( else_node );
00395         CommandNode after_else_node =  add_vertex( *graph );
00396         put(vertex_exec, *graph, after_else_node, VertexNode::normal_node );
00397         branch_stack.push( after_else_node );
00398         //branch_stack.push( build );
00399 
00400         // add edge from build to next
00401         addConditionEdge( cond, next );
00402         // add edge from build to 'after_else_node'
00403         addConditionEdge( new ConditionTrue(), else_node );
00404         proceedToNext(linenumber);
00405     }
00406 
00407     void FunctionGraphBuilder::endIfBlock(int linenumber){
00408         // this is called after a proceedToNext of the last statement of
00409         // the if block.
00410         // Connect end of if block with after_else_node
00411         CommandNode after_else_node = branch_stack.top();
00412         addConditionEdge( new ConditionTrue(), after_else_node );
00413         branch_stack.pop();
00414         // make else_node build, next remains.
00415         moveTo( branch_stack.top(), next, linenumber );
00416         branch_stack.pop();
00417         // store again !
00418         branch_stack.push( after_else_node );
00419     }
00420 
00421     // Else : can be empty and is then a plain proceed to next.
00422     void FunctionGraphBuilder::endElseBlock(int linenumber) {
00423         // after_else_node is on top of stack
00424         CommandNode after_else_node = branch_stack.top();
00425         branch_stack.pop();
00426         addConditionEdge( new ConditionTrue(), after_else_node );
00427         // make after_else_node build
00428         moveTo( after_else_node, next, linenumber );
00429     }
00430 
00431     void FunctionGraphBuilder::startWhileStatement( ConditionInterface* cond, int linenumber )
00432     {
00433         // very analogous to the if statement, but there is no else part
00434         // and we stack the first commandnode to be able to close the loop.
00435         CommandNode after_while_node = add_vertex( *graph );
00436         put(vertex_exec, *graph, after_while_node, VertexNode::normal_node );
00437         branch_stack.push( after_while_node );
00438         break_stack.push( after_while_node );
00439         branch_stack.push( build );
00440         // add edge from build to next if condition == true
00441         addConditionEdge( cond, next );
00442         // if condition fails, go from build to 'after_while_node'
00443         addConditionEdge( new ConditionTrue(), after_while_node );
00444         proceedToNext(linenumber);
00445     }
00446 
00447     void FunctionGraphBuilder::endWhileBlock(int linenumber)
00448     {
00449         CommandNode start_of_while = branch_stack.top();
00450         branch_stack.pop();
00451         // go from build back to start (and check there the condition).
00452         addConditionEdge( new ConditionTrue(), start_of_while );
00453         CommandNode after_while_node =  branch_stack.top();
00454         branch_stack.pop();
00455         break_stack.pop();
00456         moveTo( after_while_node, next, linenumber );
00457     }
00458 
00459     bool FunctionGraphBuilder::inLoop()
00460     {
00461         return break_stack.size() != 0;
00462     }
00463 
00464     bool FunctionGraphBuilder::breakLoop()
00465     {
00466         if ( !inLoop() )
00467             return false;
00468 
00469         // go from build to last nested exit point.
00470         addConditionEdge( new ConditionTrue(), break_stack.top() );
00471         return true;
00472     }
00473 
00474 #if 0
00475     void FunctionGraphBuilder::prependCommand( ActionInterface* command, int line_nr )
00476     {
00477         CommandNode previousstart = start;
00478         start = add_vertex( program );
00479         boost::property_map<Graph, vertex_exec_t>::type
00480             vmap = get(vertex_exec, program);
00481         put( vmap, start, get( vmap, previousstart ) );
00482         put( vmap, previousstart, VertexNode::normal_node );
00483         boost::property_map<Graph, vertex_command_t>::type
00484             cmap = get( vertex_command, program );
00485         VertexNode vnode( command );
00486         vnode.setLineNumber( line_nr );
00487         put( cmap, start, vnode );
00488         add_edge(start, previousstart, EdgeCondition(new ConditionTrue), program);
00489         if ( build == previousstart )
00490             build = start;
00491 
00492         // now let's reindex the vertices...
00493         boost::property_map<Graph, vertex_index_t>::type indexmap =
00494             get( vertex_index, *graph );
00495         boost::graph_traits<Graph>::vertex_iterator v, vend;
00496         int index = 0;
00497         for ( tie( v, vend ) = vertices( *graph ); v != vend; ++v )
00498             indexmap[*v] = index++;
00499     }
00500 #endif
00501 
00502 }
00503