Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/trunk/src/libraries/core/TclThreadManager.cc @ 7281

Last change on this file since 7281 was 7174, checked in by rgrieder, 14 years ago

Only catch exceptions you actually expect. And rethrow unknown exceptions ("…" can also catch internal Microsoft exceptions like floating point exception).

  • Property svn:eol-style set to native
File size: 28.6 KB
RevLine 
[1505]1/*
2 *   ORXONOX - the hottest 3D action shooter ever to exist
3 *                    > www.orxonox.net <
4 *
5 *
6 *   License notice:
7 *
8 *   This program is free software; you can redistribute it and/or
9 *   modify it under the terms of the GNU General Public License
10 *   as published by the Free Software Foundation; either version 2
11 *   of the License, or (at your option) any later version.
12 *
13 *   This program is distributed in the hope that it will be useful,
14 *   but WITHOUT ANY WARRANTY; without even the implied warranty of
15 *   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
16 *   GNU General Public License for more details.
17 *
18 *   You should have received a copy of the GNU General Public License
19 *   along with this program; if not, write to the Free Software
20 *   Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA  02110-1301, USA.
21 *
22 *   Author:
23 *      Fabian 'x3n' Landau
24 *   Co-authors:
25 *      ...
26 *
27 */
28
[1792]29#include "TclThreadManager.h"
30
[1505]31#include <boost/bind.hpp>
[3304]32#include <boost/thread/thread.hpp>
[3318]33#include <boost/thread/locks.hpp>
34#include <boost/thread/shared_mutex.hpp>
[1505]35#include <OgreTimer.h>
[3196]36#include <cpptcl/cpptcl.h>
[1505]37
[5929]38#include "util/Clock.h"
[3196]39#include "util/Convert.h"
[5747]40#include "util/Exception.h"
[6417]41#include "util/StringUtils.h"
[3196]42#include "CommandExecutor.h"
43#include "ConsoleCommand.h"
[1505]44#include "CoreIncludes.h"
45#include "TclBind.h"
[3318]46#include "TclThreadList.h"
[1505]47
48namespace orxonox
49{
[1786]50    const float TCLTHREADMANAGER_MAX_CPU_USAGE = 0.50f;
[1784]51
[1747]52    SetConsoleCommandShortcutAlias(TclThreadManager, execute, "tclexecute").argumentCompleter(0, autocompletion::tclthreads());
53    SetConsoleCommandShortcutAlias(TclThreadManager, query,   "tclquery"  ).argumentCompleter(0, autocompletion::tclthreads());
[1505]54    SetConsoleCommand(TclThreadManager, create,  false);
[1747]55    SetConsoleCommand(TclThreadManager, destroy, false).argumentCompleter(0, autocompletion::tclthreads());
56    SetConsoleCommand(TclThreadManager, execute, false).argumentCompleter(0, autocompletion::tclthreads());
57    SetConsoleCommand(TclThreadManager, query,   false).argumentCompleter(0, autocompletion::tclthreads());
[3370]58    SetConsoleCommand(TclThreadManager, source,  false).argumentCompleter(0, autocompletion::tclthreads());
[1505]59
[3318]60    /**
[5695]61        @brief A struct containing all information about a Tcl-interpreter
[3318]62    */
[3304]63    struct TclInterpreterBundle
64    {
[3318]65        TclInterpreterBundle()
66        {
67            this->lock_ = new boost::unique_lock<boost::mutex>(this->mutex_, boost::defer_lock);
68            this->bRunning_ = true;
69        }
[3304]70
[3318]71        ~TclInterpreterBundle()
72        {
73            delete this->lock_;
74        }
[3304]75
[3318]76        unsigned int                      id_;             ///< The id of the interpreter
77        Tcl::interpreter*                 interpreter_;    ///< The Tcl-interpreter
78        boost::mutex                      mutex_;          ///< A mutex to lock the interpreter while it's being used
79        boost::unique_lock<boost::mutex>* lock_;           ///< The corresponding lock for the mutex
80        TclThreadList<std::string>        queue_;          ///< The command queue for commands passed by execute(command)
81        TclThreadList<unsigned int>       queriers_;       ///< A list containing the id's of all other threads sending a query to this interpreter (to avoid circular queries and deadlocks)
82        bool                              bRunning_;       ///< This variable stays true until destroy() gets called
[3304]83    };
84
[3318]85    TclThreadManager* TclThreadManager::singletonPtr_s = 0;
[3304]86
[3318]87    /**
88        @brief Constructor
89        @param interpreter A pointer to the standard Tcl-interpreter (see @ref TclBind)
90    */
[1792]91    TclThreadManager::TclThreadManager(Tcl::interpreter* interpreter)
[1505]92    {
93        RegisterRootObject(TclThreadManager);
94
[3318]95        this->numInterpreterBundles_ = 0;
[1505]96
[3318]97        this->interpreterBundlesMutex_ = new boost::shared_mutex();
98        this->mainInterpreterMutex_ = new boost::mutex();
99        this->messageQueue_ = new TclThreadList<std::string>();
100
101        TclInterpreterBundle* newbundle = new TclInterpreterBundle();
102        newbundle->id_ = 0;
103        newbundle->interpreter_ = interpreter;
104        newbundle->lock_->lock();
105
[3313]106        {
[3318]107            boost::unique_lock<boost::shared_mutex> lock(*this->interpreterBundlesMutex_);
108            this->interpreterBundles_[0] = newbundle;
[3313]109        }
[1505]110    }
111
[3318]112    /**
113        @brief Destructor
114    */
115    TclThreadManager::~TclThreadManager()
[1505]116    {
[3318]117        delete this->interpreterBundlesMutex_;
[3326]118//        delete this->mainInterpreterMutex_; // <-- temporary disabled to avoid crash if a thread is still actively queriyng
[3318]119        delete this->messageQueue_;
[1505]120    }
121
[3318]122    /**
123        @brief The "main loop" of the TclThreadManager. Creates new threads if needed and handles queries and queued commands for the main interpreter.
124    */
[6417]125    void TclThreadManager::preUpdate(const Clock& time)
[1505]126    {
[3318]127        // Get the bundle of the main interpreter (0)
128        TclInterpreterBundle* bundle = this->getInterpreterBundle(0);
[1505]129        if (bundle)
130        {
[3318]131            // Unlock the mutex to allow other threads accessing the main interpreter
132            bundle->lock_->unlock();
133
134            // Lock the main interpreter mutex once to synchronize with threads that want to query the main interpreter
[1505]135            {
[3318]136                boost::unique_lock<boost::mutex> lock(*this->mainInterpreterMutex_);
[1505]137            }
[3318]138
139            // Lock the mutex again to gain exclusive access to the interpreter for the rest of the mainloop
140            bundle->lock_->lock();
141
142            // Execute commands in the queues of the threaded interpreters
[1505]143            {
[3318]144                boost::shared_lock<boost::shared_mutex> lock(*this->interpreterBundlesMutex_);
145                for (std::map<unsigned int, TclInterpreterBundle*>::const_iterator it = this->interpreterBundles_.begin(); it != this->interpreterBundles_.end(); ++it)
[1505]146                {
[3318]147                    if (it->first == 0)
148                        continue; // We'll handle the default interpreter later (and without threads of course)
149
150                    TclInterpreterBundle* bundle = it->second;
151                    if (!bundle->queue_.empty())
[1505]152                    {
[3318]153                        // There are commands in the queue
[1505]154                        try
155                        {
[3318]156                            if (!bundle->lock_->owns_lock() && bundle->lock_->try_lock())
[1505]157                            {
[3318]158                                // We sucessfully obtained a lock for the interpreter
159                                std::string command;
160                                if (bundle->queue_.try_pop_front(&command))
161                                {
162                                    // Start a thread to execute the command
163                                    boost::thread(boost::bind(&tclThread, bundle, command));
164                                }
165                                else
166                                {
167                                    // Somehow the queue become empty (maybe multiple consumers) - unlock the mutex
168                                    bundle->lock_->unlock();
169                                }
[1505]170                            }
[3318]171                        }
172                        catch (...)
173                        {
174                            // A lock error occurred - this is possible if the lock gets locked between !bundle->lock_->owns_lock() and bundle->lock_->try_lock()
175                            // This isn't too bad, just continue
176                        }
[1505]177                    }
178                }
179            }
180
[3318]181            // Execute commands in the message queue
182            if (!this->messageQueue_->empty())
183            {
184                std::string command;
185                while (true)
186                {
187                    // Pop the front value from the list (break the loop if there are no elements in the list)
188                    if (!this->messageQueue_->try_pop_front(&command))
189                        break;
[3313]190
[3318]191                    // Execute the command
192                    CommandExecutor::execute(command, false);
193                }
[1505]194            }
195
[3318]196            // Execute commands in the queue of the main interpreter
197            if (!bundle->queue_.empty())
[1505]198            {
[3318]199                // Calculate the time we have until we reach the maximal cpu usage
200                unsigned long maxtime = (unsigned long)(time.getDeltaTime() * 1000000 * TCLTHREADMANAGER_MAX_CPU_USAGE);
[1505]201
[3318]202                Ogre::Timer timer;
203                std::string command;
[1505]204
[3318]205                while (timer.getMicroseconds() < maxtime)
206                {
207                    // Pop the front value from the list (break the loop if there are no elements in the list)
208                    if (!bundle->queue_.try_pop_front(&command))
209                        break;
[1505]210
[3318]211                    // Execute the command
212                    CommandExecutor::execute(command, false);
213                }
214            }
[1505]215        }
216    }
217
[3318]218    /**
219        @brief Creates a new Tcl-interpreter.
220    */
221    unsigned int TclThreadManager::create()
[1505]222    {
[3318]223        TclThreadManager::getInstance().numInterpreterBundles_++;
224        TclThreadManager::createWithId(TclThreadManager::getInstance().numInterpreterBundles_);
225        COUT(0) << "Created new Tcl-interpreter with ID " << TclThreadManager::getInstance().numInterpreterBundles_ << std::endl;
226        return TclThreadManager::getInstance().numInterpreterBundles_;
[1505]227    }
228
[3318]229    /**
230        @brief Creates a new Tcl-interpreter with a given id.
[1505]231
[3318]232        Use with caution - if the id collides with an already existing interpreter, this call will fail.
233        This will also be a problem, if the auto-numbered interpreters (by using create()) reach an id
234        which was previously used in this function. Use high numbers to be safe.
235    */
236    Tcl::interpreter* TclThreadManager::createWithId(unsigned int id)
[1505]237    {
[3318]238        TclInterpreterBundle* newbundle = new TclInterpreterBundle();
239        newbundle->id_ = id;
[3370]240        newbundle->interpreter_ = TclBind::createTclInterpreter();
[1505]241
[3370]242        TclThreadManager::initialize(newbundle);
243
244        {
245            // Add the new bundle to the map
246            boost::unique_lock<boost::shared_mutex> lock(*TclThreadManager::getInstance().interpreterBundlesMutex_);
247            TclThreadManager::getInstance().interpreterBundles_[id] = newbundle;
248        }
249
250        return newbundle->interpreter_;
251    }
252
253    void TclThreadManager::initialize(TclInterpreterBundle* bundle)
254    {
[6417]255        const std::string& id_string = getConvertedValue<unsigned int, std::string>(bundle->id_);
[3370]256
[3318]257        // Initialize the new interpreter
258        try
[3313]259        {
[3318]260            // Define the functions which are implemented in C++
[3370]261            bundle->interpreter_->def("::orxonox::execute",      TclThreadManager::tcl_execute,      Tcl::variadic());
262            bundle->interpreter_->def("::orxonox::crossexecute", TclThreadManager::tcl_crossexecute, Tcl::variadic());
263            bundle->interpreter_->def("::orxonox::query",        TclThreadManager::tcl_query,        Tcl::variadic());
264            bundle->interpreter_->def("::orxonox::crossquery",   TclThreadManager::tcl_crossquery,   Tcl::variadic());
265            bundle->interpreter_->def("::orxonox::running",      TclThreadManager::tcl_running);
[3313]266
[3318]267            // Create threadspecific shortcuts for the functions above
[3370]268            bundle->interpreter_->def("execute",      TclThreadManager::tcl_execute,      Tcl::variadic());
269            bundle->interpreter_->def("crossexecute", TclThreadManager::tcl_crossexecute, Tcl::variadic());
270            bundle->interpreter_->eval("proc query      {args}    { ::orxonox::query " + id_string + " $args }");
271            bundle->interpreter_->eval("proc crossquery {id args} { ::orxonox::crossquery " + id_string + " $id $args }");
272            bundle->interpreter_->eval("proc running    {}        { return [::orxonox::running " + id_string + "] }");
[1505]273
[3318]274            // Define a variable containing the thread id
[3370]275            bundle->interpreter_->eval("set id " + id_string);
[1505]276
[3318]277            // Use our own exit function to avoid shutting down the whole program instead of just the interpreter
[3370]278            bundle->interpreter_->eval("rename exit ::tcl::exit");
279            bundle->interpreter_->eval("proc exit {} { execute TclThreadManager destroy " + id_string + " }");
[1505]280
[3318]281            // Redefine some native functions
[3370]282            bundle->interpreter_->eval("rename while ::tcl::while");
283            bundle->interpreter_->eval("rename ::orxonox::while while");
284            bundle->interpreter_->eval("rename for ::tcl::for");
285            bundle->interpreter_->eval("rename ::orxonox::for for");
[3313]286        }
[3318]287        catch (const Tcl::tcl_error& e)
[3370]288        {   bundle->interpreter_ = 0; COUT(1) << "Tcl error while creating Tcl-interpreter (" << id_string << "): " << e.what() << std::endl;   }
[3313]289    }
290
[3318]291    /**
292        @brief Stops and destroys a given Tcl-interpreter
293    */
294    void TclThreadManager::destroy(unsigned int id)
[3313]295    {
[3318]296        // TODO
297        // Not yet implemented
[3370]298        TclInterpreterBundle* bundle = TclThreadManager::getInstance().getInterpreterBundle(id);
299        if (bundle)
300        {
301            bundle->bRunning_ = false;
302        }
[3313]303    }
[1505]304
[3318]305    /**
306        @brief Sends a command to the queue of a given Tcl-interpreter
307        @param id The id of the target interpreter
308        @param command The command to be sent
309    */
310    void TclThreadManager::execute(unsigned int target_id, const std::string& command)
[3313]311    {
[3318]312        TclThreadManager::getInstance()._execute(target_id, command);
[3304]313    }
314
[3318]315    /**
316        @brief This function can be called from Tcl to execute a console command.
[3313]317
[3318]318        Commands which shall be executed are put into a queue and processed as soon as the
319        main thread feels ready to do so. The queue may also be full which results in a temporary
320        suspension of the calling thread until the queue gets ready again.
321    */
322    void TclThreadManager::tcl_execute(const Tcl::object& args)
[1505]323    {
[3318]324        TclThreadManager::getInstance()._execute(0, stripEnclosingBraces(args.get()));
[1505]325    }
326
[3318]327    /**
328        @brief This function can be called from Tcl to send a command to the queue of any interpreter.
329        @param target_id The id of the target thread
330    */
331    void TclThreadManager::tcl_crossexecute(int target_id, const Tcl::object& args)
[3313]332    {
[3318]333        TclThreadManager::getInstance()._execute(static_cast<unsigned int>(target_id), stripEnclosingBraces(args.get()));
[3313]334    }
[1505]335
[3318]336    /**
337        @brief Sends a command to the queue of a given Tcl-interpreter
338        @param id The id of the target interpreter
339        @param command The command to be sent
340    */
341    void TclThreadManager::_execute(unsigned int target_id, const std::string& command)
[1505]342    {
[3318]343        TclInterpreterBundle* bundle = this->getInterpreterBundle(target_id);
344        if (bundle)
345            bundle->queue_.push_back(command);
[1505]346    }
347
[3318]348
349    /**
350        @brief Sends a query to a given Tcl-interpreter and waits for the result
351        @param id The id of the target interpreter
352        @param command The command to be sent
353        @return The result of the command
354    */
355    std::string TclThreadManager::query(unsigned int target_id, const std::string& command)
[1505]356    {
[3318]357        return TclThreadManager::getInstance()._query(0, target_id, command);
[1505]358    }
359
[3318]360    /**
361        @brief This function can be called from Tcl to send a query to the main thread.
362        @param source_id The id of the calling thread
[3313]363
[3318]364        A query waits for the result of the command. This means, the calling thread will be blocked until
365        the main thread answers the query. In return, the main thread sends the result of the console
366        command back to Tcl.
367    */
368    std::string TclThreadManager::tcl_query(int source_id, const Tcl::object& args)
[3313]369    {
[3318]370        return TclThreadManager::getInstance()._query(static_cast<unsigned int>(source_id), 0, stripEnclosingBraces(args.get()), true);
[3313]371    }
[3307]372
[3318]373    /**
374        @brief This function can be called from Tcl to send a query to another thread.
375        @param source_id The id of the calling thread
376        @param target_id The id of the target thread
377    */
378    std::string TclThreadManager::tcl_crossquery(int source_id, int target_id, const Tcl::object& args)
[1505]379    {
[3318]380        return TclThreadManager::getInstance()._query(static_cast<unsigned int>(source_id), static_cast<unsigned int>(target_id), stripEnclosingBraces(args.get()));
[1505]381    }
382
[3318]383    /**
384        @brief This function performs a query to any Tcl interpreter
385        @param source_id The id of the calling thread
386        @param target_id The id of the target thread
387        @param command The command to send as a query
388        @param bUseCommandExecutor Only used if the target_id is 0 (which references the main interpreter). In this case it means if the command should be passed to the CommandExecutor (true) or to the main Tcl interpreter (false). This is true when called by tcl_query and false when called by tcl_crossquery.
389    */
390    std::string TclThreadManager::_query(unsigned int source_id, unsigned int target_id, const std::string& command, bool bUseCommandExecutor)
[3313]391    {
[3318]392        TclInterpreterBundle* source_bundle = this->getInterpreterBundle(source_id);
393        TclInterpreterBundle* target_bundle = this->getInterpreterBundle(target_id);
394        std::string output;
[1505]395
[3318]396        if (source_bundle && target_bundle)
[3313]397        {
[3318]398            // At this point we assume the mutex of source_bundle to be locked (because it's executing this query right now an waits for the return value)
399            // We can safely use it's querier list (because there's no other place in the code using the list except this query - and the interpreter can't start more than one query)
[3313]400
[3318]401            if ((source_bundle->id_ == target_bundle->id_) || source_bundle->queriers_.is_in(target_bundle->id_))
[3313]402            {
[3318]403                // This query would lead to a deadlock - return with an error
[6417]404                TclThreadManager::error("Error: Circular query (" + this->dumpList(source_bundle->queriers_.getList()) + ' ' + getConvertedValue<unsigned int, std::string>(source_bundle->id_) \
[3318]405                            + " -> " + getConvertedValue<unsigned int, std::string>(target_bundle->id_) \
406                            + "), couldn't query Tcl-interpreter with ID " + getConvertedValue<unsigned int, std::string>(target_bundle->id_) \
[6417]407                            + " from other interpreter with ID " + getConvertedValue<unsigned int, std::string>(source_bundle->id_) + '.');
[3313]408            }
409            else
[3318]410            {
411                boost::unique_lock<boost::mutex> lock(target_bundle->mutex_, boost::try_to_lock);
412                boost::unique_lock<boost::mutex> mainlock(*this->mainInterpreterMutex_, boost::defer_lock);
[1505]413
[3318]414                if (!lock.owns_lock() && source_bundle->id_ != 0)
[1505]415                {
[3318]416                    // We couldn't obtain the try_lock immediately and we're not the main interpreter - wait until the lock becomes possible (note: the main interpreter won't wait and instead returns an error - see below)
417                    if (target_bundle->id_ == 0)
[1505]418                    {
[3318]419                        // We're querying the main interpreter - use the main interpreter mutex to synchronize
420                        mainlock.lock();
421                        lock.lock();
422                    }
423                    else
424                    {
425                        // We're querying a threaded interpreter - no synchronization needed
426                        lock.lock();
427                    }
428                }
[1505]429
[3318]430                if (lock.owns_lock())
431                {
432                    // Now the mutex of target_bundle is also locked an we can update the querier list
433                    target_bundle->queriers_.insert(target_bundle->queriers_.getList().begin(), source_bundle->queriers_.getList().begin(), source_bundle->queriers_.getList().end());
434                    target_bundle->queriers_.push_back(source_bundle->id_);
[1505]435
[3318]436                    // Perform the query (note: this happens in the main thread because we need the returnvalue)
437                    if (target_bundle->id_ == 0 && bUseCommandExecutor)
[1505]438                    {
[3318]439                        // It's a query to the CommandExecutor
[3370]440                        TclThreadManager::debug("TclThread_query -> CE: " + command);
[3318]441                        if (!CommandExecutor::execute(command, false))
[3370]442                            TclThreadManager::error("Error: Can't execute command \"" + command + "\"!");
[3318]443
444                        if (CommandExecutor::getLastEvaluation().hasReturnvalue())
445                            output = CommandExecutor::getLastEvaluation().getReturnvalue().getString();
[1505]446                    }
447                    else
448                    {
[3318]449                        // It's a query to a Tcl interpreter
[3370]450                        TclThreadManager::debug("TclThread_query: " + command);
[3318]451
[3370]452                        output = TclThreadManager::eval(target_bundle, command, "query");
[1505]453                    }
[3318]454
455                    // Clear the queriers list of the target
456                    target_bundle->queriers_.clear();
457
458                    // Unlock the mutex of the target_bundle
459                    lock.unlock();
460
461                    // Finally unlock the main interpreter lock if necessary
462                    if (mainlock.owns_lock())
463                        mainlock.unlock();
[3313]464                }
[3318]465                else
466                {
467                    // This happens if the main thread tries to query a busy interpreter
468                    // To avoid a lock of the main thread, we simply don't proceed with the query in this case
[3370]469                    TclThreadManager::error("Error: Couldn't query Tcl-interpreter with ID " + getConvertedValue<unsigned int, std::string>(target_bundle->id_) + ", interpreter is busy right now.");
[3318]470                }
471            }
[3307]472
[1505]473        }
[3318]474
[1505]475        return output;
476    }
477
[3318]478    /**
[3370]479        @brief Creates a non-interactive Tcl-interpreter which executes a file.
480    */
481    void TclThreadManager::source(const std::string& file)
482    {
483        boost::thread(boost::bind(&sourceThread, file));
484    }
485
486    /**
[3318]487        @brief This function can be called from Tcl to ask if the thread is still suposed to be running.
488        @param id The id of the thread in question
489    */
490    bool TclThreadManager::tcl_running(int id)
[1505]491    {
[3318]492        TclInterpreterBundle* bundle = TclThreadManager::getInstance().getInterpreterBundle(static_cast<unsigned int>(id));
493        if (bundle)
494            return bundle->bRunning_;
495        else
496            return false;
497    }
498
499    /**
500        @brief Returns the interpreter bundle with the given id.
501        @param id The id of the interpreter
502        @return The interpreter or 0 if the id doesn't exist
503    */
504    TclInterpreterBundle* TclThreadManager::getInterpreterBundle(unsigned int id)
505    {
506        boost::shared_lock<boost::shared_mutex> lock(*this->interpreterBundlesMutex_);
507
508        std::map<unsigned int, TclInterpreterBundle*>::const_iterator it = this->interpreterBundles_.find(id);
509        if (it != this->interpreterBundles_.end())
[1505]510        {
[3318]511            return it->second;
[1505]512        }
[3318]513        else
[1505]514        {
[3370]515            TclThreadManager::error("Error: No Tcl-interpreter with ID " + getConvertedValue<unsigned int, std::string>(id) + " existing.");
[3318]516            return 0;
[1505]517        }
[3318]518    }
[1505]519
[3318]520    /**
521        @brief Returns a string containing all elements of a unsigned-integer-list separated by spaces.
522    */
523    std::string TclThreadManager::dumpList(const std::list<unsigned int>& list)
524    {
[6417]525        std::string output;
[3318]526        for (std::list<unsigned int>::const_iterator it = list.begin(); it != list.end(); ++it)
[1505]527        {
[3318]528            if (it != list.begin())
[6417]529                output += ' ';
[3318]530
531            output += getConvertedValue<unsigned int, std::string>(*it);
[1505]532        }
[3318]533        return output;
[1505]534    }
535
[3318]536    /**
537        @brief Returns a list with the numbers of all existing Tcl-interpreters.
538
539        This function is used by the auto completion function.
540    */
[1505]541    std::list<unsigned int> TclThreadManager::getThreadList() const
542    {
[3318]543        boost::shared_lock<boost::shared_mutex> lock(*this->interpreterBundlesMutex_);
544
[1505]545        std::list<unsigned int> threads;
546        for (std::map<unsigned int, TclInterpreterBundle*>::const_iterator it = this->interpreterBundles_.begin(); it != this->interpreterBundles_.end(); ++it)
[3318]547            if (it->first > 0 && it->first <= this->numInterpreterBundles_) // only list autonumbered interpreters (created with create()) - exclude the default interpreter 0 and all manually numbered interpreters)
548                threads.push_back(it->first);
[1505]549        return threads;
550    }
551
[3318]552    /**
553        @brief A helper function to print errors in a thread safe manner.
554    */
555    void TclThreadManager::error(const std::string& error)
[1505]556    {
[3370]557        TclThreadManager::getInstance().messageQueue_->push_back("error " + error);
[3318]558    }
559
560    /**
561        @brief A helper function to print debug information in a thread safe manner.
562    */
563    void TclThreadManager::debug(const std::string& error)
564    {
[3370]565        TclThreadManager::getInstance().messageQueue_->push_back("debug " + error);
[3318]566    }
567
568    /**
569        @brief Evaluates a Tcl command without throwing exceptions (which may rise problems on certain machines).
570        @return The Tcl return value
571
572        Errors are reported through the @ref error function.
573    */
[3370]574    std::string TclThreadManager::eval(TclInterpreterBundle* bundle, const std::string& command, const std::string& action)
[3318]575    {
576        Tcl_Interp* interpreter = bundle->interpreter_->get();
577        int cc = Tcl_Eval(interpreter, command.c_str());
578
579        Tcl::details::result result(interpreter);
580
581        if (cc != TCL_OK)
[1505]582        {
[3370]583            TclThreadManager::error("Tcl error (" + action + ", ID " + getConvertedValue<unsigned int, std::string>(bundle->id_) + "): " + static_cast<std::string>(result));
[3318]584            return "";
[1505]585        }
[3318]586        else
[1505]587        {
[3318]588            return result;
[1505]589        }
[3318]590    }
[1505]591
[3318]592    ////////////////
593    // The Thread //
594    ////////////////
595
596    /**
597        @brief The main function of the thread. Executes a Tcl command.
598        @param bundle The interpreter bundle containing all necessary variables
599        @param command the Command to execute
600    */
[6417]601    void tclThread(TclInterpreterBundle* bundle, const std::string& command)
[3318]602    {
[3370]603        TclThreadManager::debug("TclThread_execute: " + command);
[3318]604
[3370]605        TclThreadManager::eval(bundle, command, "execute");
[3318]606
607        bundle->lock_->unlock();
[1505]608    }
[3370]609
610    /**
611        @brief The main function of a non-interactive source thread. Executes the file.
612        @param file The name of the file that should be executed by the non-interactive interpreter.
613    */
[6417]614    void sourceThread(const std::string& file)
[3370]615    {
616        TclThreadManager::debug("TclThread_source: " + file);
617
618        // Prepare the command-line arguments
619        const int argc = 2;
620        char* argv[argc];
[7165]621        argv[0] = const_cast<char*>("tclthread");
[3370]622        argv[1] = const_cast<char*>(file.c_str());
623
624        // Start the Tcl-command Tcl_Main with the Tcl_OrxonoxAppInit hook
625        Tcl_Main(argc, argv, Tcl_OrxonoxAppInit);
626
627//        Tcl::object object(file);
628//        int cc = Tcl_FSEvalFile(bundle->interpreter_->get(), object.get_object());
629//        Tcl::details::result result(bundle->interpreter_->get());
630//        if (cc != TCL_OK)
631//            TclThreadManager::error("Tcl error (source, ID " + getConvertedValue<unsigned int, std::string>(bundle->id_) + "): " + static_cast<std::string>(result));
632//
633//        // Unlock the mutex
634//        bundle->lock_->unlock();
635    }
636
637    /**
638        @brief A tcl-init hook to inject the non-interactive Tcl-interpreter into the TclThreadManager.
639    */
640    int Tcl_OrxonoxAppInit(Tcl_Interp* interp)
641    {
642        // Create a new interpreter bundle
643        unsigned int id = TclThreadManager::create();
644        TclInterpreterBundle* bundle = TclThreadManager::getInstance().getInterpreterBundle(id);
645
646        // Replace the default interpreter in the bundle with the non-interactive one (passed as an argument to this function)
647        if (bundle->interpreter_)
648            delete bundle->interpreter_;
649        bundle->interpreter_ = new Tcl::interpreter(interp, true);
650
651        // Initialize the non-interactive interpreter (like in @ref TclBind::createTclInterpreter but exception safe)
[6417]652        const std::string& libpath = TclBind::getTclLibraryPath();
653        if (!libpath.empty())
654            TclThreadManager::eval(bundle, "set tcl_library \"" + libpath + '"', "source");
[3370]655        int cc = Tcl_Init(interp);
656        TclThreadManager::eval(bundle, "source \"" + TclBind::getInstance().getTclDataPath() + "/init.tcl\"", "source");
657
658        // Initialize the non-interactive interpreter also with the thread-specific stuff
659        TclThreadManager::initialize(bundle);
660
661        // Lock the mutex (this will be locked until the thread finishes - no chance to interact with the interpreter)
662        bundle->lock_->lock();
663
664        // Return to Tcl_Main
665        if (!bundle->interpreter_)
666            return TCL_ERROR;
667        else
668            return cc;
669    }
[1505]670}
Note: See TracBrowser for help on using the repository browser.