2011-01-07 19:39:27 +02:00
|
|
|
/*
|
2013-02-24 18:40:43 +01:00
|
|
|
Minetest
|
2013-02-24 19:38:45 +01:00
|
|
|
Copyright (C) 2013 celeron55, Perttu Ahola <celeron55@gmail.com>
|
2011-01-07 19:39:27 +02:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
2012-06-05 17:56:56 +03:00
|
|
|
it under the terms of the GNU Lesser General Public License as published by
|
|
|
|
the Free Software Foundation; either version 2.1 of the License, or
|
2011-01-07 19:39:27 +02:00
|
|
|
(at your option) any later version.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2012-06-05 17:56:56 +03:00
|
|
|
GNU Lesser General Public License for more details.
|
2011-01-07 19:39:27 +02:00
|
|
|
|
2012-06-05 17:56:56 +03:00
|
|
|
You should have received a copy of the GNU Lesser General Public License along
|
2011-01-07 19:39:27 +02:00
|
|
|
with this program; if not, write to the Free Software Foundation, Inc.,
|
|
|
|
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/*
|
|
|
|
Random portability stuff
|
|
|
|
|
|
|
|
See comments in porting.h
|
|
|
|
*/
|
|
|
|
|
2013-01-22 01:16:29 +04:00
|
|
|
#if defined(linux)
|
2012-12-05 21:41:05 -05:00
|
|
|
#include <unistd.h>
|
|
|
|
#elif defined(__APPLE__)
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <mach-o/dyld.h>
|
|
|
|
#elif defined(__FreeBSD__)
|
|
|
|
#include <unistd.h>
|
|
|
|
#include <sys/types.h>
|
|
|
|
#include <sys/sysctl.h>
|
|
|
|
#endif
|
|
|
|
|
2011-01-07 19:39:27 +02:00
|
|
|
#include "porting.h"
|
2011-01-08 17:34:25 +02:00
|
|
|
#include "config.h"
|
2011-02-16 22:54:07 +02:00
|
|
|
#include "debug.h"
|
2011-06-08 17:27:30 +02:00
|
|
|
#include "filesys.h"
|
2012-03-11 15:23:30 +02:00
|
|
|
#include "log.h"
|
2012-06-17 01:29:13 +03:00
|
|
|
#include "util/string.h"
|
2012-07-23 15:23:33 +03:00
|
|
|
#include <list>
|
2011-01-07 19:39:27 +02:00
|
|
|
|
2011-06-20 07:56:45 +03:00
|
|
|
#ifdef __APPLE__
|
|
|
|
#include "CoreFoundation/CoreFoundation.h"
|
|
|
|
#endif
|
|
|
|
|
2011-01-07 19:39:27 +02:00
|
|
|
namespace porting
|
|
|
|
{
|
|
|
|
|
2011-02-15 16:11:24 +02:00
|
|
|
/*
|
|
|
|
Signal handler (grabs Ctrl-C on POSIX systems)
|
|
|
|
*/
|
|
|
|
|
2011-02-15 16:12:04 +02:00
|
|
|
bool g_killed = false;
|
|
|
|
|
|
|
|
bool * signal_handler_killstatus(void)
|
|
|
|
{
|
|
|
|
return &g_killed;
|
|
|
|
}
|
|
|
|
|
2011-02-15 16:11:24 +02:00
|
|
|
#if !defined(_WIN32) // POSIX
|
|
|
|
#include <signal.h>
|
|
|
|
|
|
|
|
void sigint_handler(int sig)
|
|
|
|
{
|
|
|
|
if(g_killed == false)
|
|
|
|
{
|
2011-02-16 22:54:07 +02:00
|
|
|
dstream<<DTIME<<"INFO: sigint_handler(): "
|
2011-02-15 16:11:24 +02:00
|
|
|
<<"Ctrl-C pressed, shutting down."<<std::endl;
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2011-11-11 19:33:17 +02:00
|
|
|
// Comment out for less clutter when testing scripts
|
|
|
|
/*dstream<<DTIME<<"INFO: sigint_handler(): "
|
2011-02-16 22:54:07 +02:00
|
|
|
<<"Printing debug stacks"<<std::endl;
|
2011-11-11 19:33:17 +02:00
|
|
|
debug_stacks_print();*/
|
2011-02-16 22:54:07 +02:00
|
|
|
|
2011-02-15 16:11:24 +02:00
|
|
|
g_killed = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
(void)signal(SIGINT, SIG_DFL);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void signal_handler_init(void)
|
|
|
|
{
|
|
|
|
(void)signal(SIGINT, sigint_handler);
|
|
|
|
}
|
|
|
|
|
|
|
|
#else // _WIN32
|
2011-11-06 15:17:18 +01:00
|
|
|
#include <signal.h>
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2011-11-06 15:17:18 +01:00
|
|
|
BOOL WINAPI event_handler(DWORD sig)
|
|
|
|
{
|
|
|
|
switch(sig)
|
|
|
|
{
|
|
|
|
case CTRL_C_EVENT:
|
|
|
|
case CTRL_CLOSE_EVENT:
|
|
|
|
case CTRL_LOGOFF_EVENT:
|
|
|
|
case CTRL_SHUTDOWN_EVENT:
|
|
|
|
|
|
|
|
if(g_killed == false)
|
|
|
|
{
|
|
|
|
dstream<<DTIME<<"INFO: event_handler(): "
|
|
|
|
<<"Ctrl+C, Close Event, Logoff Event or Shutdown Event, shutting down."<<std::endl;
|
2011-11-11 19:33:17 +02:00
|
|
|
// Comment out for less clutter when testing scripts
|
|
|
|
/*dstream<<DTIME<<"INFO: event_handler(): "
|
2011-11-06 15:17:18 +01:00
|
|
|
<<"Printing debug stacks"<<std::endl;
|
2011-11-11 19:33:17 +02:00
|
|
|
debug_stacks_print();*/
|
2011-11-06 15:17:18 +01:00
|
|
|
|
|
|
|
g_killed = true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
(void)signal(SIGINT, SIG_DFL);
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
case CTRL_BREAK_EVENT:
|
|
|
|
break;
|
|
|
|
}
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2011-11-06 15:17:18 +01:00
|
|
|
return TRUE;
|
|
|
|
}
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2011-02-15 16:11:24 +02:00
|
|
|
void signal_handler_init(void)
|
|
|
|
{
|
2011-11-06 15:17:18 +01:00
|
|
|
SetConsoleCtrlHandler( (PHANDLER_ROUTINE)event_handler,TRUE);
|
2011-02-15 16:11:24 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
/*
|
|
|
|
Path mangler
|
|
|
|
*/
|
|
|
|
|
2012-03-19 23:54:56 +02:00
|
|
|
// Default to RUN_IN_PLACE style relative paths
|
|
|
|
std::string path_share = "..";
|
|
|
|
std::string path_user = "..";
|
2011-01-07 19:39:27 +02:00
|
|
|
|
2011-11-10 23:30:12 +02:00
|
|
|
std::string getDataPath(const char *subpath)
|
|
|
|
{
|
2012-03-10 15:56:24 +02:00
|
|
|
return path_share + DIR_DELIM + subpath;
|
2011-11-10 23:30:12 +02:00
|
|
|
}
|
|
|
|
|
2011-01-07 19:39:27 +02:00
|
|
|
void pathRemoveFile(char *path, char delim)
|
|
|
|
{
|
|
|
|
// Remove filename and path delimiter
|
|
|
|
int i;
|
|
|
|
for(i = strlen(path)-1; i>=0; i--)
|
|
|
|
{
|
|
|
|
if(path[i] == delim)
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
path[i] = 0;
|
|
|
|
}
|
|
|
|
|
2012-03-25 23:10:09 +03:00
|
|
|
bool detectMSVCBuildDir(char *c_path)
|
|
|
|
{
|
|
|
|
std::string path(c_path);
|
|
|
|
const char *ends[] = {"bin\\Release", "bin\\Build", NULL};
|
|
|
|
return (removeStringEnd(path, ends) != "");
|
|
|
|
}
|
|
|
|
|
2011-01-07 19:39:27 +02:00
|
|
|
void initializePaths()
|
|
|
|
{
|
2012-07-23 15:23:33 +03:00
|
|
|
#if RUN_IN_PLACE
|
2011-01-07 19:39:27 +02:00
|
|
|
/*
|
|
|
|
Use relative paths if RUN_IN_PLACE
|
|
|
|
*/
|
|
|
|
|
2012-03-11 15:23:30 +02:00
|
|
|
infostream<<"Using relative paths (RUN_IN_PLACE)"<<std::endl;
|
2011-01-07 19:39:27 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
Windows
|
|
|
|
*/
|
|
|
|
#if defined(_WIN32)
|
|
|
|
|
|
|
|
const DWORD buflen = 1000;
|
|
|
|
char buf[buflen];
|
|
|
|
DWORD len;
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2012-03-10 15:56:24 +02:00
|
|
|
// Find path of executable and set path_share relative to it
|
2011-01-07 19:39:27 +02:00
|
|
|
len = GetModuleFileName(GetModuleHandle(NULL), buf, buflen);
|
|
|
|
assert(len < buflen);
|
|
|
|
pathRemoveFile(buf, '\\');
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2012-03-25 23:10:09 +03:00
|
|
|
if(detectMSVCBuildDir(buf)){
|
|
|
|
infostream<<"MSVC build directory detected"<<std::endl;
|
|
|
|
path_share = std::string(buf) + "\\..\\..";
|
|
|
|
path_user = std::string(buf) + "\\..\\..";
|
|
|
|
}
|
|
|
|
else{
|
|
|
|
path_share = std::string(buf) + "\\..";
|
|
|
|
path_user = std::string(buf) + "\\..";
|
|
|
|
}
|
2011-01-07 19:39:27 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
Linux
|
|
|
|
*/
|
|
|
|
#elif defined(linux)
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2011-01-07 19:39:27 +02:00
|
|
|
char buf[BUFSIZ];
|
2011-02-08 01:12:55 +02:00
|
|
|
memset(buf, 0, BUFSIZ);
|
2011-01-07 19:39:27 +02:00
|
|
|
// Get path to executable
|
2011-05-16 21:57:36 +01:00
|
|
|
assert(readlink("/proc/self/exe", buf, BUFSIZ-1) != -1);
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2011-01-07 19:39:27 +02:00
|
|
|
pathRemoveFile(buf, '/');
|
|
|
|
|
2012-03-19 23:54:56 +02:00
|
|
|
path_share = std::string(buf) + "/..";
|
|
|
|
path_user = std::string(buf) + "/..";
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2011-01-07 19:39:27 +02:00
|
|
|
/*
|
|
|
|
OS X
|
|
|
|
*/
|
2012-12-05 21:41:05 -05:00
|
|
|
#elif defined(__APPLE__)
|
|
|
|
|
|
|
|
//https://developer.apple.com/library/mac/#documentation/Darwin/Reference/ManPages/man3/dyld.3.html
|
|
|
|
//TODO: Test this code
|
|
|
|
char buf[BUFSIZ];
|
|
|
|
uint32_t len = sizeof(buf);
|
2013-01-29 11:46:29 -05:00
|
|
|
assert(_NSGetExecutablePath(buf, &len) != -1);
|
2012-12-05 21:41:05 -05:00
|
|
|
|
|
|
|
pathRemoveFile(buf, '/');
|
|
|
|
|
|
|
|
path_share = std::string(buf) + "/..";
|
|
|
|
path_user = std::string(buf) + "/..";
|
|
|
|
|
|
|
|
/*
|
|
|
|
FreeBSD
|
|
|
|
*/
|
|
|
|
#elif defined(__FreeBSD__)
|
|
|
|
|
|
|
|
int mib[4];
|
|
|
|
char buf[BUFSIZ];
|
|
|
|
size_t len = sizeof(buf);
|
|
|
|
|
|
|
|
mib[0] = CTL_KERN;
|
|
|
|
mib[1] = KERN_PROC;
|
|
|
|
mib[2] = KERN_PROC_PATHNAME;
|
|
|
|
mib[3] = -1;
|
|
|
|
assert(sysctl(mib, 4, buf, &len, NULL, 0) != -1);
|
|
|
|
|
|
|
|
pathRemoveFile(buf, '/');
|
|
|
|
|
|
|
|
path_share = std::string(buf) + "/..";
|
|
|
|
path_user = std::string(buf) + "/..";
|
|
|
|
|
|
|
|
#else
|
|
|
|
|
2011-01-07 19:39:27 +02:00
|
|
|
//TODO: Get path of executable. This assumes working directory is bin/
|
2012-12-05 21:41:05 -05:00
|
|
|
dstream<<"WARNING: Relative path not properly supported on this platform"
|
2011-01-07 19:39:27 +02:00
|
|
|
<<std::endl;
|
2012-03-19 23:54:56 +02:00
|
|
|
path_share = std::string("..");
|
|
|
|
path_user = std::string("..");
|
2011-01-07 19:39:27 +02:00
|
|
|
|
|
|
|
#endif
|
2011-01-08 17:34:25 +02:00
|
|
|
|
|
|
|
#else // RUN_IN_PLACE
|
|
|
|
|
2011-01-07 19:39:27 +02:00
|
|
|
/*
|
|
|
|
Use platform-specific paths otherwise
|
|
|
|
*/
|
|
|
|
|
2012-03-11 15:23:30 +02:00
|
|
|
infostream<<"Using system-wide paths (NOT RUN_IN_PLACE)"<<std::endl;
|
2011-01-07 19:39:27 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
Windows
|
|
|
|
*/
|
|
|
|
#if defined(_WIN32)
|
|
|
|
|
|
|
|
const DWORD buflen = 1000;
|
|
|
|
char buf[buflen];
|
|
|
|
DWORD len;
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2012-03-10 15:56:24 +02:00
|
|
|
// Find path of executable and set path_share relative to it
|
2011-01-07 19:39:27 +02:00
|
|
|
len = GetModuleFileName(GetModuleHandle(NULL), buf, buflen);
|
|
|
|
assert(len < buflen);
|
|
|
|
pathRemoveFile(buf, '\\');
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2012-03-19 23:54:56 +02:00
|
|
|
// Use ".\bin\.."
|
|
|
|
path_share = std::string(buf) + "\\..";
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2011-08-10 23:43:21 +02:00
|
|
|
// Use "C:\Documents and Settings\user\Application Data\<PROJECT_NAME>"
|
2011-01-07 19:39:27 +02:00
|
|
|
len = GetEnvironmentVariable("APPDATA", buf, buflen);
|
|
|
|
assert(len < buflen);
|
2012-03-10 15:56:24 +02:00
|
|
|
path_user = std::string(buf) + DIR_DELIM + PROJECT_NAME;
|
2011-01-07 19:39:27 +02:00
|
|
|
|
|
|
|
/*
|
|
|
|
Linux
|
|
|
|
*/
|
|
|
|
#elif defined(linux)
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2011-01-10 18:21:44 +02:00
|
|
|
// Get path to executable
|
2012-07-23 15:23:33 +03:00
|
|
|
std::string bindir = "";
|
|
|
|
{
|
|
|
|
char buf[BUFSIZ];
|
|
|
|
memset(buf, 0, BUFSIZ);
|
|
|
|
assert(readlink("/proc/self/exe", buf, BUFSIZ-1) != -1);
|
|
|
|
pathRemoveFile(buf, '/');
|
|
|
|
bindir = buf;
|
|
|
|
}
|
2011-01-10 18:21:44 +02:00
|
|
|
|
2012-07-23 15:23:33 +03:00
|
|
|
// Find share directory from these.
|
|
|
|
// It is identified by containing the subdirectory "builtin".
|
|
|
|
std::list<std::string> trylist;
|
|
|
|
std::string static_sharedir = STATIC_SHAREDIR;
|
|
|
|
if(static_sharedir != "" && static_sharedir != ".")
|
|
|
|
trylist.push_back(static_sharedir);
|
|
|
|
trylist.push_back(bindir + "/../share/" + PROJECT_NAME);
|
|
|
|
trylist.push_back(bindir + "/..");
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2012-07-23 15:23:33 +03:00
|
|
|
for(std::list<std::string>::const_iterator i = trylist.begin();
|
|
|
|
i != trylist.end(); i++)
|
|
|
|
{
|
|
|
|
const std::string &trypath = *i;
|
|
|
|
if(!fs::PathExists(trypath) || !fs::PathExists(trypath + "/builtin")){
|
|
|
|
dstream<<"WARNING: system-wide share not found at \""
|
|
|
|
<<trypath<<"\""<<std::endl;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// Warn if was not the first alternative
|
|
|
|
if(i != trylist.begin()){
|
|
|
|
dstream<<"WARNING: system-wide share found at \""
|
|
|
|
<<trypath<<"\""<<std::endl;
|
|
|
|
}
|
|
|
|
path_share = trypath;
|
|
|
|
break;
|
2011-06-08 17:27:30 +02:00
|
|
|
}
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2012-03-10 15:56:24 +02:00
|
|
|
path_user = std::string(getenv("HOME")) + "/." + PROJECT_NAME;
|
2011-01-10 18:21:44 +02:00
|
|
|
|
2011-01-07 19:39:27 +02:00
|
|
|
/*
|
|
|
|
OS X
|
|
|
|
*/
|
|
|
|
#elif defined(__APPLE__)
|
2011-06-18 18:44:01 +03:00
|
|
|
|
|
|
|
// Code based on
|
|
|
|
// http://stackoverflow.com/questions/516200/relative-paths-not-working-in-xcode-c
|
|
|
|
CFBundleRef main_bundle = CFBundleGetMainBundle();
|
|
|
|
CFURLRef resources_url = CFBundleCopyResourcesDirectoryURL(main_bundle);
|
|
|
|
char path[PATH_MAX];
|
|
|
|
if(CFURLGetFileSystemRepresentation(resources_url, TRUE, (UInt8 *)path, PATH_MAX))
|
|
|
|
{
|
|
|
|
dstream<<"Bundle resource path: "<<path<<std::endl;
|
|
|
|
//chdir(path);
|
2012-03-10 15:56:24 +02:00
|
|
|
path_share = std::string(path) + "/share";
|
2011-06-18 18:44:01 +03:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
// error!
|
|
|
|
dstream<<"WARNING: Could not determine bundle resource path"<<std::endl;
|
|
|
|
}
|
|
|
|
CFRelease(resources_url);
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2012-03-10 15:56:24 +02:00
|
|
|
path_user = std::string(getenv("HOME")) + "/Library/Application Support/" + PROJECT_NAME;
|
2011-08-02 02:50:16 +04:00
|
|
|
|
|
|
|
#elif defined(__FreeBSD__)
|
|
|
|
|
2012-07-23 15:23:33 +03:00
|
|
|
path_share = STATIC_SHAREDIR;
|
2012-03-10 15:56:24 +02:00
|
|
|
path_user = std::string(getenv("HOME")) + "/." + PROJECT_NAME;
|
2012-12-05 21:41:05 -05:00
|
|
|
|
2011-01-07 19:39:27 +02:00
|
|
|
#endif
|
2011-01-08 17:34:25 +02:00
|
|
|
|
|
|
|
#endif // RUN_IN_PLACE
|
2011-01-07 19:39:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
} //namespace porting
|
|
|
|
|