Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/branches/libraries/src/core/Core.cc @ 5625

Last change on this file since 5625 was 5625, checked in by rgrieder, 15 years ago

Added plugin path that gets configured in BuildConfig.cmake and correctly set in Core.

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