X-Git-Url: http://git.tdb.fi/?p=libs%2Fcore.git;a=blobdiff_plain;f=source%2Fdir.cpp;h=29c24c188fa96b7d76763b9b1ab5ad4a8aa3de53;hp=9d23a1388015a89acf8e9ce36cf257c4fd8941a7;hb=9c48f72f02e3112f4193cc8b3f12b135aa483de1;hpb=a17d6ad286e19e2222ab8b6c9a762a83bf2c6c56 diff --git a/source/dir.cpp b/source/dir.cpp index 9d23a13..29c24c1 100644 --- a/source/dir.cpp +++ b/source/dir.cpp @@ -14,6 +14,7 @@ Distributed under the LGPL #endif #include #include +#include #include "dir.h" #include "path.h" #include "stat.h" @@ -24,6 +25,44 @@ using namespace std; namespace Msp { namespace FS { +namespace +{ + +/** +Helper function to determine the location of the program's executable. Caches +the last result to cut down filesystem access with repeated calls. +*/ +const Path &get_bin_dir(const string &argv0) +{ + static string last_argv0; + static Path bin_dir; + + if(!(argv0==last_argv0)) + { + Path exe; + if(argv0.find('/')==string::npos) + { + const char *path=getenv("PATH"); + vector dirs=split(path, ':'); + for(vector::const_iterator i=dirs.begin(); i!=dirs.end(); ++i) + if(exists(Path(*i)/argv0)) + { + exe=realpath(Path(*i)/argv0); + break; + } + } + else + exe=realpath(argv0); + + last_argv0=argv0; + bin_dir=dirname(exe); + } + + return bin_dir; +} + +} + void mkdir(const Path &path, int mode) { int err; @@ -46,7 +85,7 @@ void mkpath(const Path &path, int mode) { p/=*i; #ifdef WIN32 - if(p.size()==1 && is_windows_drive(*i)) + if(p.size()==1 && p.is_absolute()) continue; #endif struct stat st; @@ -141,10 +180,35 @@ Path get_user_data_dir(const string &appname) return Path(datadir)/appname; return "."; #else - return get_home_dir()/("+"+appname); + return get_home_dir()/("."+appname); #endif } +Path get_sys_conf_dir(const string &argv0) +{ + Path dir=get_bin_dir(argv0); + + if(dir[-1]=="bin" || dir[-1]=="sbin") + { + dir/=".."; + if(dir[-1]=="usr") + dir/=".."; + return dir/"etc"; + } + else + return dir; +} + +Path get_sys_data_dir(const string &argv0, const string &appname) +{ + Path dir=get_bin_dir(argv0); + + if(dir[-1]=="bin" || dir[-1]=="sbin") + return dir/".."/"share"/appname; + else + return dir; +} + void chdir(const Path &path) { if(::chdir(path.str().c_str())==-1)