Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/branches/resource2/src/core/Core.cc @ 5654

Last change on this file since 5654 was 5654, checked in by rgrieder, 15 years ago
  • Implemented file management via resource manager and loading of resource locations via XML. Changes made:
    • SoundManager loads via memory stream rather than via file
    • Loader uses LuaState::includeFile() to load an XML file and passes the lua tag remover function to its LuaState.
    • ConfigFileManager still loads with hard paths because the files are required before Ogre gets created
  • Renamed LuaBind to LuaState, deSingletonised it and added new features:
    • doFile(), doString(), includeFile(), includeString() where include will preparse the string with a function provided with LuaState::setIncludeParser
    • Moved lua tags replace function to Loader (since it's actually an XML related task)
    • Using data_path/lua/LuaInitScript.lua to provide the following functions
      • logMessage(level, message)
      • doFile, dofile, include (all working with relative paths but within the same resource group)
  • Modified Script class to work with LuaState and fixed its XML Loader
  • Adjusted all level and include files (both "include" and "dofile" lua commands)
  • Property svn:eol-style set to native
File size: 25.5 KB
Line 
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 *      Reto Grieder
25 *   Co-authors:
26 *      ...
27 *
28 */
29
30/**
31@file
32@brief
33    Implementation of the Core singleton with its global variables (avoids boost include)
34*/
35
36#include "Core.h"
37
38#include <cassert>
39#include <fstream>
40#include <cstdlib>
41#include <cstdio>
42#include <boost/filesystem.hpp>
43#include <OgreRenderWindow.h>
44
45#ifdef ORXONOX_PLATFORM_WINDOWS
46#  ifndef WIN32_LEAN_AND_MEAN
47#    define WIN32_LEAN_AND_MEAN
48#  endif
49#  include <windows.h>
50#  undef min
51#  undef max
52#elif defined(ORXONOX_PLATFORM_APPLE)
53#  include <sys/param.h>
54#  include <mach-o/dyld.h>
55#else /* Linux */
56#  include <sys/types.h>
57#  include <unistd.h>
58#endif
59
60#include "SpecialConfig.h"
61#include "util/Debug.h"
62#include "util/Exception.h"
63#include "util/SignalHandler.h"
64#include "Clock.h"
65#include "CommandExecutor.h"
66#include "CommandLine.h"
67#include "ConfigFileManager.h"
68#include "ConfigValueIncludes.h"
69#include "CoreIncludes.h"
70#include "Factory.h"
71#include "GameMode.h"
72#include "GraphicsManager.h"
73#include "GUIManager.h"
74#include "Identifier.h"
75#include "Language.h"
76#include "LuaState.h"
77#include "Shell.h"
78#include "TclBind.h"
79#include "TclThreadManager.h"
80#include "ToluaBindCore.h"
81#include "input/InputManager.h"
82
83namespace orxonox
84{
85    //! Static pointer to the singleton
86    Core* Core::singletonPtr_s  = 0;
87
88    SetCommandLineArgument(externalDataPath, "").information("Path to the external data files");
89    SetCommandLineOnlyArgument(writingPathSuffix, "").information("Additional subfolder for config and log files");
90    SetCommandLineArgument(settingsFile, "orxonox.ini").information("THE configuration file");
91#ifdef ORXONOX_PLATFORM_WINDOWS
92    SetCommandLineArgument(limitToCPU, 0).information("Limits the program to one cpu/core (1, 2, 3, etc.). 0 turns it off (default)");
93#endif
94
95    /**
96    @brief
97        Helper class for the Core singleton: we cannot derive
98        Core from OrxonoxClass because we need to handle the Identifier
99        destruction in the Core destructor.
100    */
101    class CoreConfiguration : public OrxonoxClass
102    {
103    public:
104        CoreConfiguration()
105        {
106        }
107
108        void initialise()
109        {
110            RegisterRootObject(CoreConfiguration);
111            this->setConfigValues();
112
113            // External data directory only exists for dev runs
114            if (Core::isDevelopmentRun())
115            {
116                // Possible data path override by the command line
117                if (!CommandLine::getArgument("externalDataPath")->hasDefaultValue())
118                    tsetExternalDataPath(CommandLine::getValue("externalDataPath"));
119            }
120        }
121
122        /**
123            @brief Function to collect the SetConfigValue-macro calls.
124        */
125        void setConfigValues()
126        {
127#ifdef NDEBUG
128            const unsigned int defaultLevelConsole = 1;
129            const unsigned int defaultLevelLogfile = 3;
130            const unsigned int defaultLevelShell   = 1;
131#else
132            const unsigned int defaultLevelConsole = 3;
133            const unsigned int defaultLevelLogfile = 4;
134            const unsigned int defaultLevelShell   = 3;
135#endif
136            SetConfigValue(softDebugLevelConsole_, defaultLevelConsole)
137                .description("The maximal level of debug output shown in the console")
138                .callback(this, &CoreConfiguration::debugLevelChanged);
139            SetConfigValue(softDebugLevelLogfile_, defaultLevelLogfile)
140                .description("The maximal level of debug output shown in the logfile")
141                .callback(this, &CoreConfiguration::debugLevelChanged);
142            SetConfigValue(softDebugLevelShell_, defaultLevelShell)
143                .description("The maximal level of debug output shown in the ingame shell")
144                .callback(this, &CoreConfiguration::debugLevelChanged);
145
146            SetConfigValue(language_, Language::getInstance().defaultLanguage_)
147                .description("The language of the ingame text")
148                .callback(this, &CoreConfiguration::languageChanged);
149            SetConfigValue(bInitializeRandomNumberGenerator_, true)
150                .description("If true, all random actions are different each time you start the game")
151                .callback(this, &CoreConfiguration::initializeRandomNumberGenerator);
152        }
153
154        /**
155            @brief Callback function if the debug level has changed.
156        */
157        void debugLevelChanged()
158        {
159            // softDebugLevel_ is the maximum of the 3 variables
160            this->softDebugLevel_ = this->softDebugLevelConsole_;
161            if (this->softDebugLevelLogfile_ > this->softDebugLevel_)
162                this->softDebugLevel_ = this->softDebugLevelLogfile_;
163            if (this->softDebugLevelShell_ > this->softDebugLevel_)
164                this->softDebugLevel_ = this->softDebugLevelShell_;
165
166            OutputHandler::setSoftDebugLevel(OutputHandler::LD_All,     this->softDebugLevel_);
167            OutputHandler::setSoftDebugLevel(OutputHandler::LD_Console, this->softDebugLevelConsole_);
168            OutputHandler::setSoftDebugLevel(OutputHandler::LD_Logfile, this->softDebugLevelLogfile_);
169            OutputHandler::setSoftDebugLevel(OutputHandler::LD_Shell,   this->softDebugLevelShell_);
170        }
171
172        /**
173            @brief Callback function if the language has changed.
174        */
175        void languageChanged()
176        {
177            // Read the translation file after the language was configured
178            Language::getInstance().readTranslatedLanguageFile();
179        }
180
181        /**
182            @brief Sets the language in the config-file back to the default.
183        */
184        void resetLanguage()
185        {
186            ResetConfigValue(language_);
187        }
188
189        /**
190        @brief
191            Temporary sets the data path
192        @param path
193            The new data path
194        */
195        void tsetExternalDataPath(const std::string& path)
196        {
197            dataPath_ = boost::filesystem::path(path);
198        }
199
200        void initializeRandomNumberGenerator()
201        {
202            static bool bInitialized = false;
203            if (!bInitialized && this->bInitializeRandomNumberGenerator_)
204            {
205                srand(static_cast<unsigned int>(time(0)));
206                rand();
207                bInitialized = true;
208            }
209        }
210
211        int softDebugLevel_;                            //!< The debug level
212        int softDebugLevelConsole_;                     //!< The debug level for the console
213        int softDebugLevelLogfile_;                     //!< The debug level for the logfile
214        int softDebugLevelShell_;                       //!< The debug level for the ingame shell
215        std::string language_;                          //!< The language
216        bool bInitializeRandomNumberGenerator_;         //!< If true, srand(time(0)) is called
217
218        //! Path to the parent directory of the ones above if program was installed with relativ pahts
219        boost::filesystem::path rootPath_;
220        boost::filesystem::path executablePath_;        //!< Path to the executable
221        boost::filesystem::path dataPath_;              //!< Path to the data file folder
222        boost::filesystem::path externalDataPath_;      //!< Path to the external data file folder
223        boost::filesystem::path configPath_;            //!< Path to the config file folder
224        boost::filesystem::path logPath_;               //!< Path to the log file folder
225    };
226
227
228    Core::Core(const std::string& cmdLine)
229        // Cleanup guard for identifier destruction (incl. XMLPort, configValues, consoleCommands)
230        : identifierDestroyer_(Identifier::destroyAllIdentifiers)
231        // Cleanup guard for external console commands that don't belong to an Identifier
232        , consoleCommandDestroyer_(CommandExecutor::destroyExternalCommands)
233        , configuration_(new CoreConfiguration()) // Don't yet create config values!
234        , bDevRun_(false)
235        , bGraphicsLoaded_(false)
236    {
237        // Parse command line arguments first
238        CommandLine::parseCommandLine(cmdLine);
239
240        // Determine and set the location of the executable
241        setExecutablePath();
242
243        // Determine whether we have an installed or a binary dir run
244        // The latter occurs when simply running from the build directory
245        checkDevBuild();
246
247        // Make sure the directories we write in exist or else make them
248        createDirectories();
249
250        // create a signal handler (only active for linux)
251        // This call is placed as soon as possible, but after the directories are set
252        this->signalHandler_.reset(new SignalHandler());
253        this->signalHandler_->doCatch(configuration_->executablePath_.string(), Core::getLogPathString() + "orxonox_crash.log");
254
255        // Set the correct log path. Before this call, /tmp (Unix) or %TEMP% was used
256        OutputHandler::getOutStream().setLogPath(Core::getLogPathString());
257
258        // Parse additional options file now that we know its path
259        CommandLine::parseFile();
260
261#ifdef ORXONOX_PLATFORM_WINDOWS
262        // limit the main thread to the first core so that QueryPerformanceCounter doesn't jump
263        // do this after ogre has initialised. Somehow Ogre changes the settings again (not through
264        // the timer though).
265        int limitToCPU = CommandLine::getValue("limitToCPU");
266        if (limitToCPU > 0)
267            setThreadAffinity(static_cast<unsigned int>(limitToCPU));
268#endif
269        // Add tolua interface
270        LuaState::addToluaInterface(&tolua_Core_open, "Core");
271
272        // Manage ini files and set the default settings file (usually orxonox.ini)
273        this->configFileManager_.reset(new ConfigFileManager());
274        this->configFileManager_->setFilename(ConfigFileType::Settings,
275            CommandLine::getValue("settingsFile").getString());
276
277        // Required as well for the config values
278        this->languageInstance_.reset(new Language());
279
280        // creates the class hierarchy for all classes with factories
281        Factory::createClassHierarchy();
282
283        // Do this soon after the ConfigFileManager has been created to open up the
284        // possibility to configure everything below here
285        this->configuration_->initialise();
286
287        // Load OGRE excluding the renderer and the render window
288        this->graphicsManager_.reset(new GraphicsManager(false));
289
290        // initialise Tcl
291        this->tclBind_.reset(new TclBind(Core::getDataPathString()));
292        this->tclThreadManager_.reset(new TclThreadManager(tclBind_->getTclInterpreter()));
293
294        // create a shell
295        this->shell_.reset(new Shell());
296    }
297
298    /**
299    @brief
300        All destruction code is handled by scoped_ptrs and ScopeGuards.
301    */
302    Core::~Core()
303    {
304    }
305
306    void Core::loadGraphics()
307    {
308        if (bGraphicsLoaded_)
309            return;
310
311        // Upgrade OGRE to receive a render window
312        graphicsManager_->upgradeToGraphics();
313
314        // The render window width and height are used to set up the mouse movement.
315        size_t windowHnd = 0;
316        graphicsManager_->getRenderWindow()->getCustomAttribute("WINDOW", &windowHnd);
317
318        // Calls the InputManager which sets up the input devices.
319        scoped_ptr<InputManager> inputManager(new InputManager(windowHnd));
320
321        // load the CEGUI interface
322        guiManager_.reset(new GUIManager(graphicsManager_->getRenderWindow()));
323
324        // Dismiss scoped pointer
325        inputManager_.swap(inputManager);
326
327        bGraphicsLoaded_ = true;
328    }
329
330    void Core::unloadGraphics()
331    {
332        if (!bGraphicsLoaded_)
333            return;
334
335        this->guiManager_.reset();;
336        this->inputManager_.reset();;
337        this->graphicsManager_.reset();
338
339        // Load Ogre::Root again, but without the render system
340        this->graphicsManager_.reset(new GraphicsManager(false));
341
342        bGraphicsLoaded_ = false;
343    }
344
345    /**
346        @brief Returns the softDebugLevel for the given device (returns a default-value if the class is right about to be created).
347        @param device The device
348        @return The softDebugLevel
349    */
350    /*static*/ int Core::getSoftDebugLevel(OutputHandler::OutputDevice device)
351    {
352        switch (device)
353        {
354        case OutputHandler::LD_All:
355            return Core::getInstance().configuration_->softDebugLevel_;
356        case OutputHandler::LD_Console:
357            return Core::getInstance().configuration_->softDebugLevelConsole_;
358        case OutputHandler::LD_Logfile:
359            return Core::getInstance().configuration_->softDebugLevelLogfile_;
360        case OutputHandler::LD_Shell:
361            return Core::getInstance().configuration_->softDebugLevelShell_;
362        default:
363            assert(0);
364            return 2;
365        }
366    }
367
368     /**
369        @brief Sets the softDebugLevel for the given device. Please use this only temporary and restore the value afterwards, as it overrides the configured value.
370        @param device The device
371        @param level The level
372    */
373    /*static*/ void Core::setSoftDebugLevel(OutputHandler::OutputDevice device, int level)
374    {
375        if (device == OutputHandler::LD_All)
376            Core::getInstance().configuration_->softDebugLevel_ = level;
377        else if (device == OutputHandler::LD_Console)
378            Core::getInstance().configuration_->softDebugLevelConsole_ = level;
379        else if (device == OutputHandler::LD_Logfile)
380            Core::getInstance().configuration_->softDebugLevelLogfile_ = level;
381        else if (device == OutputHandler::LD_Shell)
382            Core::getInstance().configuration_->softDebugLevelShell_ = level;
383
384        OutputHandler::setSoftDebugLevel(device, level);
385    }
386
387    /**
388        @brief Returns the configured language.
389    */
390    /*static*/ const std::string& Core::getLanguage()
391    {
392        return Core::getInstance().configuration_->language_;
393    }
394
395    /**
396        @brief Sets the language in the config-file back to the default.
397    */
398    /*static*/ void Core::resetLanguage()
399    {
400        Core::getInstance().configuration_->resetLanguage();
401    }
402
403    /*static*/ void Core::tsetExternalDataPath(const std::string& path)
404    {
405        getInstance().configuration_->tsetExternalDataPath(path);
406    }
407
408    /*static*/ const boost::filesystem::path& Core::getDataPath()
409    {
410        return getInstance().configuration_->dataPath_;
411    }
412    /*static*/ std::string Core::getDataPathString()
413    {
414        return getInstance().configuration_->dataPath_.string() + '/';
415    }
416
417    /*static*/ const boost::filesystem::path& Core::getExternalDataPath()
418    {
419        return getInstance().configuration_->externalDataPath_;
420    }
421    /*static*/ std::string Core::getExternalDataPathString()
422    {
423        return getInstance().configuration_->externalDataPath_.string() + '/';
424    }
425
426    /*static*/ const boost::filesystem::path& Core::getConfigPath()
427    {
428        return getInstance().configuration_->configPath_;
429    }
430    /*static*/ std::string Core::getConfigPathString()
431    {
432        return getInstance().configuration_->configPath_.string() + '/';
433    }
434
435    /*static*/ const boost::filesystem::path& Core::getLogPath()
436    {
437        return getInstance().configuration_->logPath_;
438    }
439    /*static*/ std::string Core::getLogPathString()
440    {
441        return getInstance().configuration_->logPath_.string() + '/';
442    }
443
444    /*static*/ const boost::filesystem::path& Core::getRootPath()
445    {
446        return getInstance().configuration_->rootPath_;
447    }
448    /*static*/ std::string Core::getRootPathString()
449    {
450        return getInstance().configuration_->rootPath_.string() + '/';
451    }
452
453    /**
454    @note
455        The code of this function has been copied and adjusted from OGRE, an open source graphics engine.
456            (Object-oriented Graphics Rendering Engine)
457        For the latest info, see http://www.ogre3d.org/
458
459        Copyright (c) 2000-2008 Torus Knot Software Ltd
460
461        OGRE is licensed under the LGPL. For more info, see OGRE license.
462    */
463    void Core::setThreadAffinity(int limitToCPU)
464    {
465#ifdef ORXONOX_PLATFORM_WINDOWS
466
467        if (limitToCPU <= 0)
468            return;
469
470        unsigned int coreNr = limitToCPU - 1;
471        // Get the current process core mask
472        DWORD procMask;
473        DWORD sysMask;
474#  if _MSC_VER >= 1400 && defined (_M_X64)
475        GetProcessAffinityMask(GetCurrentProcess(), (PDWORD_PTR)&procMask, (PDWORD_PTR)&sysMask);
476#  else
477        GetProcessAffinityMask(GetCurrentProcess(), &procMask, &sysMask);
478#  endif
479
480        // If procMask is 0, consider there is only one core available
481        // (using 0 as procMask will cause an infinite loop below)
482        if (procMask == 0)
483            procMask = 1;
484
485        // if the core specified with coreNr is not available, take the lowest one
486        if (!(procMask & (1 << coreNr)))
487            coreNr = 0;
488
489        // Find the lowest core that this process uses and coreNr suggests
490        DWORD threadMask = 1;
491        while ((threadMask & procMask) == 0 || (threadMask < (1u << coreNr)))
492            threadMask <<= 1;
493
494        // Set affinity to the first core
495        SetThreadAffinityMask(GetCurrentThread(), threadMask);
496#endif
497    }
498
499    /**
500    @brief
501        Compares the executable path with the working directory
502    */
503    void Core::setExecutablePath()
504    {
505#ifdef ORXONOX_PLATFORM_WINDOWS
506        // get executable module
507        TCHAR buffer[1024];
508        if (GetModuleFileName(NULL, buffer, 1024) == 0)
509            ThrowException(General, "Could not retrieve executable path.");
510
511#elif defined(ORXONOX_PLATFORM_APPLE)
512        char buffer[1024];
513        unsigned long path_len = 1023;
514        if (_NSGetExecutablePath(buffer, &path_len))
515            ThrowException(General, "Could not retrieve executable path.");
516
517#else /* Linux */
518        /* written by Nicolai Haehnle <prefect_@gmx.net> */
519
520        /* Get our PID and build the name of the link in /proc */
521        char linkname[64]; /* /proc/<pid>/exe */
522        if (snprintf(linkname, sizeof(linkname), "/proc/%i/exe", getpid()) < 0)
523        {
524            /* This should only happen on large word systems. I'm not sure
525               what the proper response is here.
526               Since it really is an assert-like condition, aborting the
527               program seems to be in order. */
528            assert(false);
529        }
530
531        /* Now read the symbolic link */
532        char buffer[1024];
533        int ret;
534        ret = readlink(linkname, buffer, 1024);
535        /* In case of an error, leave the handling up to the caller */
536        if (ret == -1)
537            ThrowException(General, "Could not retrieve executable path.");
538
539        /* Ensure proper NUL termination */
540        buffer[ret] = 0;
541#endif
542
543        configuration_->executablePath_ = boost::filesystem::path(buffer);
544#ifndef ORXONOX_PLATFORM_APPLE
545        configuration_->executablePath_ = configuration_->executablePath_.branch_path(); // remove executable name
546#endif
547    }
548
549    /**
550    @brief
551        Checks for "orxonox_dev_build.keep_me" in the executable diretory.
552        If found it means that this is not an installed run, hence we
553        don't write the logs and config files to ~/.orxonox
554    @throws
555        GeneralException
556    */
557    void Core::checkDevBuild()
558    {
559        if (boost::filesystem::exists(configuration_->executablePath_ / "orxonox_dev_build.keep_me"))
560        {
561            COUT(1) << "Running from the build tree." << std::endl;
562            Core::bDevRun_ = true;
563            configuration_->dataPath_   = specialConfig::dataDevDirectory;
564            configuration_->externalDataPath_ = specialConfig::externalDataDevDirectory;
565            configuration_->configPath_ = specialConfig::configDevDirectory;
566            configuration_->logPath_    = specialConfig::logDevDirectory;
567        }
568        else
569        {
570#ifdef INSTALL_COPYABLE // --> relative paths
571            // Also set the root path
572            boost::filesystem::path relativeExecutablePath(specialConfig::defaultRuntimePath);
573            configuration_->rootPath_ = configuration_->executablePath_;
574            while (!boost::filesystem::equivalent(configuration_->rootPath_ / relativeExecutablePath, configuration_->executablePath_)
575                   && !configuration_->rootPath_.empty())
576                configuration_->rootPath_ = configuration_->rootPath_.branch_path();
577            if (configuration_->rootPath_.empty())
578                ThrowException(General, "Could not derive a root directory. Might the binary installation directory contain '..' when taken relative to the installation prefix path?");
579
580            // Using paths relative to the install prefix, complete them
581            configuration_->dataPath_   = configuration_->rootPath_ / specialConfig::defaultDataPath;
582            configuration_->configPath_ = configuration_->rootPath_ / specialConfig::defaultConfigPath;
583            configuration_->logPath_    = configuration_->rootPath_ / specialConfig::defaultLogPath;
584#else
585            // There is no root path, so don't set it at all
586
587            configuration_->dataPath_  = specialConfig::dataInstallDirectory;
588
589            // Get user directory
590#  ifdef ORXONOX_PLATFORM_UNIX /* Apple? */
591            char* userDataPathPtr(getenv("HOME"));
592#  else
593            char* userDataPathPtr(getenv("APPDATA"));
594#  endif
595            if (userDataPathPtr == NULL)
596                ThrowException(General, "Could not retrieve user data path.");
597            boost::filesystem::path userDataPath(userDataPathPtr);
598            userDataPath /= ".orxonox";
599
600            configuration_->configPath_ = userDataPath / specialConfig::defaultConfigPath;
601            configuration_->logPath_    = userDataPath / specialConfig::defaultLogPath;
602#endif
603        }
604
605        // Option to put all the config and log files in a separate folder
606        if (!CommandLine::getArgument("writingPathSuffix")->hasDefaultValue())
607        {
608            std::string directory(CommandLine::getValue("writingPathSuffix").getString());
609            configuration_->configPath_ = configuration_->configPath_ / directory;
610            configuration_->logPath_    = configuration_->logPath_    / directory;
611        }
612    }
613
614    /*
615    @brief
616        Checks for the log and the config directory and creates them
617        if necessary. Otherwise me might have problems opening those files.
618    @throws
619        orxonox::GeneralException if the directory to be created is a file.
620    */
621    void Core::createDirectories()
622    {
623        std::vector<std::pair<boost::filesystem::path, std::string> > directories;
624        directories.push_back(std::make_pair(boost::filesystem::path(configuration_->configPath_), "config"));
625        directories.push_back(std::make_pair(boost::filesystem::path(configuration_->logPath_), "log"));
626
627        for (std::vector<std::pair<boost::filesystem::path, std::string> >::iterator it = directories.begin();
628            it != directories.end(); ++it)
629        {
630            if (boost::filesystem::exists(it->first) && !boost::filesystem::is_directory(it->first))
631            {
632                ThrowException(General, std::string("The ") + it->second + " directory has been preoccupied by a file! \
633                                         Please remove " + it->first.string());
634            }
635            if (boost::filesystem::create_directories(it->first)) // function may not return true at all (bug?)
636            {
637                COUT(4) << "Created " << it->second << " directory" << std::endl;
638            }
639        }
640    }
641
642    bool Core::preUpdate(const Clock& time) throw()
643    {
644        std::string exceptionMessage;
645        try
646        {
647            if (this->bGraphicsLoaded_)
648            {
649                // process input events
650                this->inputManager_->update(time);
651                // process gui events
652                this->guiManager_->update(time);
653            }
654            // process thread commands
655            this->tclThreadManager_->update(time);
656        }
657        catch (const std::exception& ex)
658        { exceptionMessage = ex.what(); }
659        catch (...)
660        { exceptionMessage = "Unknown exception"; }
661        if (!exceptionMessage.empty())
662        {
663            COUT(0) << "An exception occurred in the Core preUpdate: " << exceptionMessage << std::endl;
664            COUT(0) << "This should really never happen! Closing the program." << std::endl;
665            return false;
666        }
667        return true;
668    }
669
670    bool Core::postUpdate(const Clock& time) throw()
671    {
672        std::string exceptionMessage;
673        try
674        {
675            if (this->bGraphicsLoaded_)
676            {
677                // Render (doesn't throw)
678                this->graphicsManager_->update(time);
679            }
680        }
681        catch (const std::exception& ex)
682        { exceptionMessage = ex.what(); }
683        catch (...)
684        { exceptionMessage = "Unknown exception"; }
685        if (!exceptionMessage.empty())
686        {
687            COUT(0) << "An exception occurred in the Core postUpdate: " << exceptionMessage << std::endl;
688            COUT(0) << "This should really never happen! Closing the program." << std::endl;
689            return false;
690        }
691        return true;
692    }
693}
Note: See TracBrowser for help on using the repository browser.