Planet
navi homePPSaboutscreenshotsdownloaddevelopmentforum

source: code/trunk/src/core/Core.cc @ 2871

Last change on this file since 2871 was 2759, checked in by scheusso, 16 years ago

merged network branch (windows,multiplayer fixes) back to trunk

  • Property svn:eol-style set to native
File size: 15.9 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:
[1755]25 *      Reto Grieder
[1505]26 *
27 */
28
29/**
[2171]30    @file
[1524]31    @brief Implementation of the Core class.
[1505]32*/
33
[1524]34#include "Core.h"
[2710]35
[1756]36#include <cassert>
[2710]37#include <fstream>
38#include <cstdlib>
39#include <cstdio>
40#include <boost/filesystem.hpp>
41
42#ifdef ORXONOX_PLATFORM_WINDOWS
43#  include <windows.h>
44#elif defined(ORXONOX_PLATFORM_APPLE)
45#  include <sys/param.h>
46#  include <mach-o/dyld.h>
47#else /* Linux */
48#  include <sys/types.h>
49#  include <unistd.h>
50#endif
51
52#include "SpecialConfig.h"
53#include "util/Exception.h"
[1505]54#include "Language.h"
55#include "CoreIncludes.h"
56#include "ConfigValueIncludes.h"
[2710]57#include "LuaBind.h"
58#include "CommandLine.h"
[1505]59
60namespace orxonox
61{
[2710]62    //! Path to the parent directory of the ones above if program was installed with relativ pahts
63    static boost::filesystem::path rootPath_g;
64    static boost::filesystem::path executablePath_g;            //!< Path to the executable
65    static boost::filesystem::path mediaPath_g;                 //!< Path to the media file folder
66    static boost::filesystem::path configPath_g;                //!< Path to the config file folder
67    static boost::filesystem::path logPath_g;                   //!< Path to the log file folder
68
[2087]69    bool Core::bShowsGraphics_s = false;
70    bool Core::bHasServer_s     = false;
71    bool Core::bIsClient_s      = false;
72    bool Core::bIsStandalone_s  = false;
73    bool Core::bIsMaster_s      = false;
74
[2710]75    bool Core::isDevBuild_s     = false;
76    Core* Core::singletonRef_s  = 0;
[2662]77
[2710]78    SetCommandLineArgument(mediaPath, "").information("PATH");
79    SetCommandLineArgument(directory, "").information("DIR");
80
[1505]81    /**
82        @brief Constructor: Registers the object and sets the config-values.
83        @param A reference to a global variable, used to avoid an infinite recursion in getSoftDebugLevel()
84    */
[1524]85    Core::Core()
[1505]86    {
[1524]87        RegisterRootObject(Core);
[2662]88
89        assert(Core::singletonRef_s == 0);
90        Core::singletonRef_s = this;
[2710]91
[2662]92        this->bInitializeRandomNumberGenerator_ = false;
[2710]93        this->setConfigValues();
[2662]94
[2710]95        // Set the correct log path. Before this call, /tmp (Unix) or %TEMP% was used
96        OutputHandler::getOutStream().setLogPath(Core::getLogPathString());
97
98        // Possible media path override by the command line
99        if (!CommandLine::getArgument("mediaPath")->hasDefaultValue())
100        {
101            //std::string mediaPath = CommandLine::getValue("mediaPath");
102            Core::tsetMediaPath(CommandLine::getValue("mediaPath"));
103        }
[1505]104    }
105
106    /**
107        @brief Sets the bool to true to avoid static functions accessing a deleted object.
108    */
[1524]109    Core::~Core()
[1505]110    {
[2662]111        assert(Core::singletonRef_s);
112        Core::singletonRef_s = 0;
[1505]113    }
114
115    /**
116        @brief Function to collect the SetConfigValue-macro calls.
117    */
[1524]118    void Core::setConfigValues()
[1505]119    {
[2710]120#ifdef NDEBUG
121        const unsigned int defaultLevelConsole = 1;
122        const unsigned int defaultLevelLogfile = 3;
123        const unsigned int defaultLevelShell   = 1;
124#else
125        const unsigned int defaultLevelConsole = 3;
126        const unsigned int defaultLevelLogfile = 4;
127        const unsigned int defaultLevelShell   = 3;
128#endif
129        SetConfigValue(softDebugLevelConsole_, defaultLevelConsole)
130            .description("The maximal level of debug output shown in the console").callback(this, &Core::debugLevelChanged);
131        SetConfigValue(softDebugLevelLogfile_, defaultLevelLogfile)
132            .description("The maximal level of debug output shown in the logfile").callback(this, &Core::debugLevelChanged);
133        SetConfigValue(softDebugLevelShell_, defaultLevelShell)
134            .description("The maximal level of debug output shown in the ingame shell").callback(this, &Core::debugLevelChanged);
135
[1747]136        SetConfigValue(language_, Language::getLanguage().defaultLanguage_).description("The language of the ingame text").callback(this, &Core::languageChanged);
[2662]137        SetConfigValue(bInitializeRandomNumberGenerator_, true).description("If true, all random actions are different each time you start the game").callback(this, &Core::initializeRandomNumberGenerator);
[2710]138
[2727]139        SetConfigValue(mediaPathString_, mediaPath_g.string())
[2710]140            .description("Relative path to the game data.").callback(this, &Core::mediaPathChanged);
[1747]141    }
[1505]142
[1747]143    /**
144        @brief Callback function if the debug level has changed.
145    */
146    void Core::debugLevelChanged()
147    {
[1505]148        // softDebugLevel_ is the maximum of the 3 variables
149        this->softDebugLevel_ = this->softDebugLevelConsole_;
150        if (this->softDebugLevelLogfile_ > this->softDebugLevel_)
151            this->softDebugLevel_ = this->softDebugLevelLogfile_;
152        if (this->softDebugLevelShell_ > this->softDebugLevel_)
153            this->softDebugLevel_ = this->softDebugLevelShell_;
154
[1747]155        OutputHandler::setSoftDebugLevel(OutputHandler::LD_All,     this->softDebugLevel_);
156        OutputHandler::setSoftDebugLevel(OutputHandler::LD_Console, this->softDebugLevelConsole_);
157        OutputHandler::setSoftDebugLevel(OutputHandler::LD_Logfile, this->softDebugLevelLogfile_);
158        OutputHandler::setSoftDebugLevel(OutputHandler::LD_Shell,   this->softDebugLevelShell_);
159    }
[1505]160
[1747]161    /**
162        @brief Callback function if the language has changed.
163    */
164    void Core::languageChanged()
165    {
166        // Read the translation file after the language was configured
167        Language::getLanguage().readTranslatedLanguageFile();
[1505]168    }
169
170    /**
[2710]171    @brief
172        Callback function if the media path has changed.
173    */
174    void Core::mediaPathChanged()
175    {
176        mediaPath_g = boost::filesystem::path(this->mediaPathString_);
177    }
178
179    /**
[1505]180        @brief Returns the softDebugLevel for the given device (returns a default-value if the class ist right about to be created).
181        @param device The device
182        @return The softDebugLevel
183    */
[1524]184    int Core::getSoftDebugLevel(OutputHandler::OutputDevice device)
[1505]185    {
[2662]186        switch (device)
[1505]187        {
[2662]188        case OutputHandler::LD_All:
189            return Core::getInstance().softDebugLevel_;
190        case OutputHandler::LD_Console:
191            return Core::getInstance().softDebugLevelConsole_;
192        case OutputHandler::LD_Logfile:
193            return Core::getInstance().softDebugLevelLogfile_;
194        case OutputHandler::LD_Shell:
195            return Core::getInstance().softDebugLevelShell_;
196        default:
197            assert(0);
198            return 2;
[1505]199        }
200    }
201
202     /**
203        @brief Sets the softDebugLevel for the given device. Please use this only temporary and restore the value afterwards, as it overrides the configured value.
204        @param device The device
205        @param level The level
206    */
[1524]207     void Core::setSoftDebugLevel(OutputHandler::OutputDevice device, int level)
[1505]208     {
[2662]209        if (device == OutputHandler::LD_All)
210            Core::getInstance().softDebugLevel_ = level;
211        else if (device == OutputHandler::LD_Console)
212            Core::getInstance().softDebugLevelConsole_ = level;
213        else if (device == OutputHandler::LD_Logfile)
214            Core::getInstance().softDebugLevelLogfile_ = level;
215        else if (device == OutputHandler::LD_Shell)
216            Core::getInstance().softDebugLevelShell_ = level;
[1747]217
[2662]218        OutputHandler::setSoftDebugLevel(device, level);
[1505]219     }
220
221    /**
222        @brief Returns the configured language.
223    */
[1524]224    const std::string& Core::getLanguage()
[1505]225    {
[2662]226        return Core::getInstance().language_;
[1505]227    }
228
229    /**
230        @brief Sets the language in the config-file back to the default.
231    */
[1524]232    void Core::resetLanguage()
[1505]233    {
[1524]234        Core::getInstance().resetLanguageIntern();
[1505]235    }
236
237    /**
238        @brief Sets the language in the config-file back to the default.
239    */
[1524]240    void Core::resetLanguageIntern()
[1505]241    {
242        ResetConfigValue(language_);
243    }
[2662]244
[2710]245    /**
246    @brief
247        Temporary sets the media path
248    @param path
249        The new media path
250    */
251    void Core::_tsetMediaPath(const std::string& path)
252    {
253        ModifyConfigValue(mediaPathString_, tset, path);
254    }
255
256    /*static*/ const boost::filesystem::path& Core::getMediaPath()
257    {
258        return mediaPath_g;
259    }
260    /*static*/ std::string Core::getMediaPathString()
261    {
262        return mediaPath_g.string() + '/';
263    }
264
265    /*static*/ const boost::filesystem::path& Core::getConfigPath()
266    {
267        return configPath_g;
268    }
269    /*static*/ std::string Core::getConfigPathString()
270    {
271        return configPath_g.string() + '/';
272    }
273
274    /*static*/ const boost::filesystem::path& Core::getLogPath()
275    {
276        return logPath_g;
277    }
278    /*static*/ std::string Core::getLogPathString()
279    {
280        return logPath_g.string() + '/';
281    }
282
[2662]283    void Core::initializeRandomNumberGenerator()
284    {
285        static bool bInitialized = false;
286        if (!bInitialized && this->bInitializeRandomNumberGenerator_)
287        {
288            srand(time(0));
289            rand();
290            bInitialized = true;
291        }
292    }
[2710]293
294    /**
295    @brief
296        Performs the rather lower level operations just after
297        int main() has been called.
298    @remarks
299        This gets called AFTER pre-main stuff like AddFactory,
300        SetConsoleCommand, etc.
301    */
302    /*static*/ void Core::postMainInitialisation()
303    {
304        // set location of the executable
305        Core::setExecutablePath();
306
307        // Determine whether we have an installed or a binary dir run
308        // The latter occurs when simply running from the build directory
309        Core::checkDevBuild();
310
311        // Make sure the directories we write in exist or else make them
312        Core::createDirectories();
313    }
314
315    /**
316    @brief
317        Compares the executable path with the working directory
318    */
319    /*static*/ void Core::setExecutablePath()
320    {
321#ifdef ORXONOX_PLATFORM_WINDOWS
322        // get executable module
323        TCHAR buffer[1024];
324        if (GetModuleFileName(NULL, buffer, 1024) == 0)
325            ThrowException(General, "Could not retrieve executable path.");
326
327#elif defined(ORXONOX_PLATFORM_APPLE)
328        char buffer[1024];
329        unsigned long path_len = 1023;
330        if (_NSGetExecutablePath(buffer, &path_len))
331            ThrowException(General, "Could not retrieve executable path.");
332
333#else /* Linux */
334        /* written by Nicolai Haehnle <prefect_@gmx.net> */
335
336        /* Get our PID and build the name of the link in /proc */
337        char linkname[64]; /* /proc/<pid>/exe */
338        if (snprintf(linkname, sizeof(linkname), "/proc/%i/exe", getpid()) < 0)
339        {
340            /* This should only happen on large word systems. I'm not sure
341               what the proper response is here.
342               Since it really is an assert-like condition, aborting the
343               program seems to be in order. */
344            assert(false);
345        }
346
347        /* Now read the symbolic link */
348        char buffer[1024];
349        int ret;
350        ret = readlink(linkname, buffer, 1024);
351        /* In case of an error, leave the handling up to the caller */
352        if (ret == -1)
353            ThrowException(General, "Could not retrieve executable path.");
354
355        /* Ensure proper NUL termination */
356        buffer[ret] = 0;
357#endif
358
359        executablePath_g = boost::filesystem::path(buffer);
360#ifndef ORXONOX_PLATFORM_APPLE
361        executablePath_g = executablePath_g.branch_path(); // remove executable name
362#endif
363    }
364
365    /**
366    @brief
367        Checks for "orxonox_dev_build.keep_me" in the executable diretory.
368        If found it means that this is not an installed run, hence we
369        don't write the logs and config files to ~/.orxonox
370    */
371    /*static*/ void Core::checkDevBuild()
372    {
373        if (boost::filesystem::exists(executablePath_g / "orxonox_dev_build.keep_me"))
374        {
375            COUT(1) << "Running from the build tree." << std::endl;
376            Core::isDevBuild_s = true;
377            mediaPath_g  = ORXONOX_MEDIA_DEV_PATH;
378            configPath_g = ORXONOX_CONFIG_DEV_PATH;
379            logPath_g    = ORXONOX_LOG_DEV_PATH;
380        }
381        else
382        {
383#ifdef INSTALL_COPYABLE // --> relative paths
384            // Also set the root path
385            boost::filesystem::path relativeExecutablePath(ORXONOX_RUNTIME_INSTALL_PATH);
386            rootPath_g = executablePath_g;
387            while (!boost::filesystem::equivalent(rootPath_g / relativeExecutablePath, executablePath_g) || rootPath_g.empty())
388                rootPath_g = rootPath_g.branch_path();
389            if (rootPath_g.empty())
390                ThrowException(General, "Could not derive a root directory. Might the binary installation directory contain '..' when taken relative to the installation prefix path?");
391
392            // Using paths relative to the install prefix, complete them
393            mediaPath_g  = rootPath_g / ORXONOX_MEDIA_INSTALL_PATH;
394            configPath_g = rootPath_g / ORXONOX_CONFIG_INSTALL_PATH;
395            logPath_g    = rootPath_g / ORXONOX_LOG_INSTALL_PATH;
396#else
397            // There is no root path, so don't set it at all
398
399            mediaPath_g  = ORXONOX_MEDIA_INSTALL_PATH;
400
401            // Get user directory
402#  ifdef ORXONOX_PLATFORM_UNIX /* Apple? */
403            char* userDataPathPtr(getenv("HOME"));
404#  else
405            char* userDataPathPtr(getenv("APPDATA"));
406#  endif
407            if (userDataPathPtr == NULL)
408                ThrowException(General, "Could not retrieve user data path.");
409            boost::filesystem::path userDataPath(userDataPathPtr);
410            userDataPath /= ".orxonox";
411
412            configPath_g = userDataPath / ORXONOX_CONFIG_INSTALL_PATH;
413            logPath_g    = userDataPath / ORXONOX_LOG_INSTALL_PATH;
414#endif
415        }
416
417        // Option to put all the config and log files in a separate folder
418        if (!CommandLine::getArgument("directory")->hasDefaultValue())
419        {
[2712]420            std::string directory(CommandLine::getValue("directory").getString());
[2710]421            configPath_g = configPath_g / directory;
422            logPath_g    = logPath_g    / directory;
423        }
424    }
425
426    /*
427    @brief
428        Checks for the log and the config directory and creates them
429        if necessary. Otherwise me might have problems opening those files.
430    */
431    /*static*/ void Core::createDirectories()
432    {
433        std::vector<std::pair<boost::filesystem::path, std::string> > directories;
434        directories.push_back(std::pair<boost::filesystem::path, std::string>
435            (boost::filesystem::path(configPath_g), "config"));
436        directories.push_back(std::pair<boost::filesystem::path, std::string>
437            (boost::filesystem::path(logPath_g),    "log"));
438
439        for (std::vector<std::pair<boost::filesystem::path, std::string> >::iterator it = directories.begin();
440            it != directories.end(); ++it)
441        {
442            if (boost::filesystem::exists(it->first) && !boost::filesystem::is_directory(it->first))
443            {
444                ThrowException(General, std::string("The ") + it->second + " directory has been preoccupied by a file! \
[2759]445                                         Please remove " + it->first.string());
[2710]446            }
447            if (boost::filesystem::create_directories(it->first)) // function may not return true at all (bug?)
448            {
449                COUT(4) << "Created " << it->second << " directory" << std::endl;
450            }
451        }
452    }
[1505]453}
Note: See TracBrowser for help on using the repository browser.