You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
1341 lines
28 KiB
C++
1341 lines
28 KiB
C++
// NeL - MMORPG Framework <http://dev.ryzom.com/projects/nel/>
|
|
// Copyright (C) 2010 Winch Gate Property Limited
|
|
//
|
|
// This program is free software: you can redistribute it and/or modify
|
|
// it under the terms of the GNU Affero General Public License as
|
|
// published by the Free Software Foundation, either version 3 of the
|
|
// License, or (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
|
|
// GNU Affero General Public License for more details.
|
|
//
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
// along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
#include "stdmisc.h"
|
|
|
|
#include "nel/misc/types_nl.h"
|
|
#include "nel/misc/common.h"
|
|
|
|
#ifdef NL_OS_WINDOWS
|
|
# include <ShellAPI.h>
|
|
# include <io.h>
|
|
# include <tchar.h>
|
|
|
|
#define popen _popen
|
|
#define pclose _pclose
|
|
|
|
#elif defined NL_OS_MAC
|
|
# include <ApplicationServices/ApplicationServices.h>
|
|
#elif defined NL_OS_UNIX
|
|
# include <unistd.h>
|
|
# include <cerrno>
|
|
# include <pthread.h>
|
|
# include <sched.h>
|
|
#endif
|
|
|
|
#define MAX_LINE_WIDTH 256
|
|
|
|
#include "nel/misc/command.h"
|
|
#include "nel/misc/path.h"
|
|
#include "nel/misc/i18n.h"
|
|
|
|
using namespace std;
|
|
|
|
#ifndef NL_COMP_MINGW
|
|
#ifdef NL_OS_WINDOWS
|
|
# pragma message( " " )
|
|
|
|
# if FINAL_VERSION
|
|
# pragma message( "************************" )
|
|
# pragma message( "**** FINAL_VERSION *****" )
|
|
# pragma message( "************************" )
|
|
# else
|
|
# pragma message( "Not using FINAL_VERSION")
|
|
# endif // FINAL_VERSION
|
|
|
|
# ifdef ASSERT_THROW_EXCEPTION
|
|
# pragma message( "nlassert throws exceptions" )
|
|
# else
|
|
# pragma message( "nlassert does not throw exceptions" )
|
|
# endif // ASSERT_THROW_EXCEPTION
|
|
|
|
# ifdef _STLPORT_VERSION
|
|
# pragma message( "Using STLport" )
|
|
# else
|
|
# pragma message( "Using standard STL" )
|
|
# endif // _STLPORT_VERSION
|
|
|
|
# pragma message( " " )
|
|
|
|
# if (_MSC_VER >= 1200) && (_MSC_VER < 1400) && (WINVER < 0x0500)
|
|
//Using VC7 and later lib, need this to compile on VC6
|
|
extern "C" long _ftol2( double dblSource ) { return _ftol( dblSource ); }
|
|
# endif
|
|
|
|
|
|
#endif // NL_OS_WINDOWS
|
|
#endif // !NL_COMP_MINGW
|
|
|
|
|
|
#if defined(NL_HAS_SSE2) && !defined(NL_CPU_X86_64)
|
|
|
|
void *operator new(size_t size) throw(std::bad_alloc)
|
|
{
|
|
void *p = aligned_malloc(size, NL_DEFAULT_MEMORY_ALIGNMENT);
|
|
if (p == NULL) throw std::bad_alloc();
|
|
return p;
|
|
}
|
|
|
|
void *operator new[](size_t size) throw(std::bad_alloc)
|
|
{
|
|
void *p = aligned_malloc(size, NL_DEFAULT_MEMORY_ALIGNMENT);
|
|
if (p == NULL) throw std::bad_alloc();
|
|
return p;
|
|
}
|
|
|
|
void operator delete(void *p) throw()
|
|
{
|
|
aligned_free(p);
|
|
}
|
|
|
|
void operator delete[](void *p) throw()
|
|
{
|
|
aligned_free(p);
|
|
}
|
|
|
|
#endif /* NL_HAS_SSE2 */
|
|
|
|
|
|
#ifdef DEBUG_NEW
|
|
#define new DEBUG_NEW
|
|
#endif
|
|
|
|
namespace NLMISC
|
|
{
|
|
|
|
/*
|
|
* Portable Sleep() function that suspends the execution of the calling thread for a number of milliseconds.
|
|
* Note: the resolution of the timer is system-dependant and may be more than 1 millisecond.
|
|
*/
|
|
void nlSleep( uint32 ms )
|
|
{
|
|
#ifdef NL_OS_WINDOWS
|
|
|
|
#ifdef NL_DEBUG
|
|
// a Sleep(0) "block" the other thread in DEBUG/_CONSOLE, so we clamp
|
|
ms = max(ms, (uint32)1);
|
|
#endif
|
|
|
|
Sleep( ms );
|
|
|
|
#elif defined NL_OS_UNIX
|
|
//usleep( ms*1000 ); // resolution: 20 ms!
|
|
|
|
timespec ts;
|
|
ts.tv_sec = ms/1000;
|
|
ts.tv_nsec = (ms%1000)*1000000;
|
|
int res;
|
|
do
|
|
{
|
|
res = nanosleep( &ts, &ts ); // resolution: 10 ms (with common scheduling policy)
|
|
}
|
|
while ( (res != 0) && (errno==EINTR) );
|
|
#endif
|
|
}
|
|
|
|
|
|
/*
|
|
* Returns Thread Id (note: on Linux, Process Id is the same as the Thread Id)
|
|
*/
|
|
size_t getThreadId()
|
|
{
|
|
#ifdef NL_OS_WINDOWS
|
|
return GetCurrentThreadId();
|
|
#elif defined NL_OS_UNIX
|
|
return size_t(pthread_self());
|
|
// doesnt work on linux kernel 2.6 return getpid();
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
/*
|
|
* Returns a readable string from a vector of bytes. '\0' are replaced by ' '
|
|
*/
|
|
string stringFromVector( const vector<uint8>& v, bool limited )
|
|
{
|
|
string s;
|
|
|
|
if (!v.empty())
|
|
{
|
|
int size = (int)v.size ();
|
|
if (limited && size > 1000)
|
|
{
|
|
string middle = "...<buf too big,skip middle part>...";
|
|
s.resize (1000 + middle.size());
|
|
memcpy (&*s.begin(), &*v.begin(), 500);
|
|
memcpy (&*s.begin()+500, &*middle.begin(), middle.size());
|
|
memcpy (&*s.begin()+500+middle.size(), &*v.begin()+size-500, 500);
|
|
}
|
|
else
|
|
{
|
|
s.resize (size);
|
|
memcpy( &*s.begin(), &*v.begin(), v.size() );
|
|
}
|
|
|
|
// Replace '\0' characters
|
|
string::iterator is;
|
|
for ( is=s.begin(); is!=s.end(); ++is )
|
|
{
|
|
// remplace non printable char and % with '?' chat
|
|
if ( ! isprint((uint8)(*is)) || (*is) == '%')
|
|
{
|
|
(*is) = '?';
|
|
}
|
|
}
|
|
}
|
|
/*
|
|
if ( ! v.empty() )
|
|
{
|
|
// Copy contents
|
|
s.resize( v.size() );
|
|
memcpy( &*s.begin(), &*v.begin(), v.size() );
|
|
|
|
// Replace '\0' characters
|
|
string::iterator is;
|
|
for ( is=s.begin(); is!=s.end(); ++is )
|
|
{
|
|
// remplace non printable char and % with '?' chat
|
|
if ( ! isprint((*is)) || (*is) == '%')
|
|
{
|
|
(*is) = '?';
|
|
}
|
|
}
|
|
}
|
|
*/ return s;
|
|
}
|
|
|
|
|
|
sint smprintf( char *buffer, size_t count, const char *format, ... )
|
|
{
|
|
sint ret;
|
|
|
|
va_list args;
|
|
va_start( args, format );
|
|
ret = vsnprintf( buffer, count, format, args );
|
|
if ( ret == -1 )
|
|
{
|
|
buffer[count-1] = '\0';
|
|
}
|
|
va_end( args );
|
|
|
|
return( ret );
|
|
}
|
|
|
|
|
|
sint64 atoiInt64 (const char *ident, sint64 base)
|
|
{
|
|
sint64 number = 0;
|
|
bool neg = false;
|
|
|
|
// NULL string
|
|
nlassert (ident != NULL);
|
|
|
|
// empty string
|
|
if (*ident == '\0') goto end;
|
|
|
|
// + sign
|
|
if (*ident == '+') ident++;
|
|
|
|
// - sign
|
|
if (*ident == '-') { neg = true; ident++; }
|
|
|
|
while (*ident != '\0')
|
|
{
|
|
if (isdigit((unsigned char)*ident))
|
|
{
|
|
number *= base;
|
|
number += (*ident)-'0';
|
|
}
|
|
else if (base > 10 && islower((unsigned char)*ident))
|
|
{
|
|
number *= base;
|
|
number += (*ident)-'a'+10;
|
|
}
|
|
else if (base > 10 && isupper((unsigned char)*ident))
|
|
{
|
|
number *= base;
|
|
number += (*ident)-'A'+10;
|
|
}
|
|
else
|
|
{
|
|
goto end;
|
|
}
|
|
ident++;
|
|
}
|
|
end:
|
|
if (neg) number = -number;
|
|
return number;
|
|
}
|
|
|
|
void itoaInt64 (sint64 number, char *str, sint64 base)
|
|
{
|
|
str[0] = '\0';
|
|
char b[256];
|
|
if(!number)
|
|
{
|
|
str[0] = '0';
|
|
str[1] = '\0';
|
|
return;
|
|
}
|
|
memset(b,'\0',255);
|
|
memset(b,'0',64);
|
|
sint n;
|
|
sint64 x = number;
|
|
if (x < 0) x = -x;
|
|
char baseTable[] = "0123456789abcdefghijklmnopqrstuvwyz";
|
|
for(n = 0; n < 64; n ++)
|
|
{
|
|
sint num = (sint)(x % base);
|
|
b[64 - n] = baseTable[num];
|
|
if(!x)
|
|
{
|
|
int k;
|
|
int j = 0;
|
|
|
|
if (number < 0)
|
|
{
|
|
str[j++] = '-';
|
|
}
|
|
|
|
for(k = 64 - n + 1; k <= 64; k++)
|
|
{
|
|
str[j ++] = b[k];
|
|
}
|
|
str[j] = '\0';
|
|
break;
|
|
}
|
|
x /= base;
|
|
}
|
|
}
|
|
|
|
uint raiseToNextPowerOf2(uint v)
|
|
{
|
|
uint res=1;
|
|
while(res<v)
|
|
res<<=1;
|
|
|
|
return res;
|
|
}
|
|
|
|
uint getPowerOf2(uint v)
|
|
{
|
|
uint res=1;
|
|
uint ret=0;
|
|
while(res<v)
|
|
{
|
|
ret++;
|
|
res<<=1;
|
|
}
|
|
|
|
return ret;
|
|
}
|
|
|
|
bool isPowerOf2(sint32 v)
|
|
{
|
|
while(v)
|
|
{
|
|
if(v&1)
|
|
{
|
|
v>>=1;
|
|
if(v)
|
|
return false;
|
|
}
|
|
else
|
|
v>>=1;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
|
|
string bytesToHumanReadable (const std::string &bytes)
|
|
{
|
|
return bytesToHumanReadable (atoiInt64(bytes.c_str()));
|
|
}
|
|
|
|
string bytesToHumanReadable (uint64 bytes)
|
|
{
|
|
static const char *divTable[]= { "B", "KiB", "MiB", "GiB", "TiB" };
|
|
uint div = 0;
|
|
uint64 res = bytes;
|
|
uint64 newres = res;
|
|
for(;;)
|
|
{
|
|
newres /= 1024;
|
|
if(newres < 8 || div > 3)
|
|
break;
|
|
div++;
|
|
res = newres;
|
|
}
|
|
return toString ("%" NL_I64 "u%s", res, divTable[div]);
|
|
}
|
|
|
|
uint32 humanReadableToBytes (const string &str)
|
|
{
|
|
uint32 res;
|
|
|
|
if(str.empty())
|
|
return 0;
|
|
|
|
// not a number
|
|
if(str[0]<'0' || str[0]>'9')
|
|
return 0;
|
|
|
|
res = atoi (str.c_str());
|
|
|
|
if(str[str.size()-1] == 'B')
|
|
{
|
|
if (str.size()<3)
|
|
return res;
|
|
|
|
// there's no break and it's **normal**
|
|
switch (str[str.size()-2])
|
|
{
|
|
case 'G': res *= 1024;
|
|
case 'M': res *= 1024;
|
|
case 'K': res *= 1024;
|
|
default: ;
|
|
}
|
|
}
|
|
|
|
return res;
|
|
}
|
|
|
|
|
|
NLMISC_CATEGORISED_COMMAND(nel,btohr, "Convert a bytes number into an human readable number", "<int>")
|
|
{
|
|
nlunreferenced(rawCommandString);
|
|
nlunreferenced(quiet);
|
|
nlunreferenced(human);
|
|
|
|
if (args.size() != 1)
|
|
return false;
|
|
|
|
log.displayNL("%s -> %s", args[0].c_str(), bytesToHumanReadable(args[0]).c_str());
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
NLMISC_CATEGORISED_COMMAND(nel,hrtob, "Convert a human readable number into a bytes number", "<hr>")
|
|
{
|
|
nlunreferenced(rawCommandString);
|
|
nlunreferenced(quiet);
|
|
nlunreferenced(human);
|
|
|
|
if (args.size() != 1)
|
|
return false;
|
|
|
|
log.displayNL("%s -> %u", args[0].c_str(), humanReadableToBytes(args[0]));
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
string secondsToHumanReadable (uint32 time)
|
|
{
|
|
static const char *divTable[] = { "s", "mn", "h", "d" };
|
|
static uint divCoef[] = { 60, 60, 24 };
|
|
uint div = 0;
|
|
uint32 res = time;
|
|
uint32 newres = res;
|
|
for(;;)
|
|
{
|
|
if(div > 2)
|
|
break;
|
|
|
|
newres /= divCoef[div];
|
|
|
|
if(newres < 3)
|
|
break;
|
|
|
|
div++;
|
|
res = newres;
|
|
}
|
|
return toString ("%u%s", res, divTable[div]);
|
|
}
|
|
|
|
uint32 fromHumanReadable (const std::string &str)
|
|
{
|
|
if (str.size() == 0)
|
|
return 0;
|
|
|
|
uint32 val;
|
|
fromString(str, val);
|
|
|
|
switch (str[str.size()-1])
|
|
{
|
|
case 's': return val; // second
|
|
case 'n': return val*60; // minutes (mn)
|
|
case 'h': return val*60*60; // hour
|
|
case 'd': return val*60*60*24; // day
|
|
case 'b': // bytes
|
|
switch (str[str.size()-2])
|
|
{
|
|
case 'k': return val*1024;
|
|
case 'm': return val*1024*1024;
|
|
case 'g': return val*1024*1024*1024;
|
|
default : return val;
|
|
}
|
|
default: return val;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
NLMISC_CATEGORISED_COMMAND(nel,stohr, "Convert a second number into an human readable time", "<int>")
|
|
{
|
|
nlunreferenced(rawCommandString);
|
|
nlunreferenced(quiet);
|
|
nlunreferenced(human);
|
|
|
|
if (args.size() != 1)
|
|
return false;
|
|
|
|
uint32 seconds;
|
|
fromString(args[0], seconds);
|
|
log.displayNL("%s -> %s", args[0].c_str(), secondsToHumanReadable(seconds).c_str());
|
|
|
|
return true;
|
|
}
|
|
|
|
|
|
std::string toLower(const std::string &str)
|
|
{
|
|
string res;
|
|
res.reserve(str.size());
|
|
for(uint i = 0; i < str.size(); i++)
|
|
{
|
|
if( (str[i] >= 'A') && (str[i] <= 'Z') )
|
|
res += str[i] - 'A' + 'a';
|
|
else
|
|
res += str[i];
|
|
}
|
|
return res;
|
|
}
|
|
|
|
char toLower(const char ch)
|
|
{
|
|
if( (ch >= 'A') && (ch <= 'Z') )
|
|
{
|
|
return ch - 'A' + 'a';
|
|
}
|
|
else
|
|
{
|
|
return ch;
|
|
}
|
|
}
|
|
|
|
void toLower(char *str)
|
|
{
|
|
if (str == 0)
|
|
return;
|
|
|
|
while(*str != '\0')
|
|
{
|
|
if( (*str >= 'A') && (*str <= 'Z') )
|
|
{
|
|
*str = *str - 'A' + 'a';
|
|
}
|
|
str++;
|
|
}
|
|
}
|
|
|
|
std::string toUpper(const std::string &str)
|
|
{
|
|
string res;
|
|
res.reserve(str.size());
|
|
for(uint i = 0; i < str.size(); i++)
|
|
{
|
|
if( (str[i] >= 'a') && (str[i] <= 'z') )
|
|
res += str[i] - 'a' + 'A';
|
|
else
|
|
res += str[i];
|
|
}
|
|
return res;
|
|
}
|
|
|
|
void toUpper(char *str)
|
|
{
|
|
if (str == 0)
|
|
return;
|
|
|
|
while(*str != '\0')
|
|
{
|
|
if( (*str >= 'a') && (*str <= 'z') )
|
|
{
|
|
*str = *str - 'a' + 'A';
|
|
}
|
|
str++;
|
|
}
|
|
}
|
|
|
|
std::string formatThousands(const std::string& s)
|
|
{
|
|
sint i, k;
|
|
sint remaining = (sint)s.length() - 1;
|
|
static std::string separator = NLMISC::CI18N::get("uiThousandsSeparator").toUtf8();
|
|
|
|
// Don't add separator if the number is < 10k
|
|
if (remaining < 4) return s;
|
|
|
|
std::string ns;
|
|
|
|
do
|
|
{
|
|
for (i = remaining, k = 0; i >= 0 && k < 3; --i, ++k )
|
|
{
|
|
ns = s[i] + ns; // New char is added to front of ns
|
|
if ( i > 0 && k == 2) ns = separator + ns; // j > 0 means still more digits
|
|
}
|
|
|
|
remaining -= 3;
|
|
}
|
|
while (remaining >= 0);
|
|
|
|
return ns;
|
|
}
|
|
|
|
//
|
|
// Exceptions
|
|
//
|
|
|
|
Exception::Exception() : _Reason("Unknown Exception")
|
|
{
|
|
// nlinfo("Exception will be launched: %s", _Reason.c_str());
|
|
}
|
|
|
|
Exception::Exception(const std::string &reason) : _Reason(reason)
|
|
{
|
|
nlinfo("Exception will be launched: %s", _Reason.c_str());
|
|
}
|
|
|
|
Exception::Exception(const char *format, ...)
|
|
{
|
|
NLMISC_CONVERT_VARGS (_Reason, format, NLMISC::MaxCStringSize);
|
|
nlinfo("Exception will be launched: %s", _Reason.c_str());
|
|
}
|
|
|
|
const char *Exception::what() const throw()
|
|
{
|
|
return _Reason.c_str();
|
|
}
|
|
|
|
bool killProgram(uint32 pid)
|
|
{
|
|
#ifdef NL_OS_UNIX
|
|
int res = kill(pid, SIGKILL);
|
|
if(res == -1)
|
|
{
|
|
char *err = strerror (errno);
|
|
nlwarning("Failed to kill '%d' err %d: '%s'", pid, errno, err);
|
|
}
|
|
return res == 0;
|
|
/*#elif defined(NL_OS_WINDOWS)
|
|
// it doesn't work because pid != handle and i don't know how to kill a pid or know the real handle of another service (not -1)
|
|
int res = TerminateProcess((HANDLE)pid, 888);
|
|
LPVOID lpMsgBuf;
|
|
FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMsgBuf, 0, NULL);
|
|
nlwarning("Failed to kill '%d' err %d: '%s'", pid, GetLastError (), lpMsgBuf);
|
|
LocalFree(lpMsgBuf);
|
|
return res != 0;
|
|
*/
|
|
#else
|
|
nlwarning("kill not implemented on this OS");
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
bool abortProgram(uint32 pid)
|
|
{
|
|
#ifdef NL_OS_UNIX
|
|
int res = kill(pid, SIGABRT);
|
|
if(res == -1)
|
|
{
|
|
char *err = strerror (errno);
|
|
nlwarning("Failed to abort '%d' err %d: '%s'", pid, errno, err);
|
|
}
|
|
return res == 0;
|
|
#else
|
|
nlwarning("abort not implemented on this OS");
|
|
return false;
|
|
#endif
|
|
}
|
|
|
|
bool launchProgram(const std::string &programName, const std::string &arguments, bool log)
|
|
{
|
|
#ifdef NL_OS_WINDOWS
|
|
STARTUPINFOA si;
|
|
PROCESS_INFORMATION pi;
|
|
|
|
memset(&si, 0, sizeof(si));
|
|
memset(&pi, 0, sizeof(pi));
|
|
|
|
si.cb = sizeof(si);
|
|
|
|
// Enable nlassert/nlstop to display the error reason & callstack
|
|
const TCHAR *SE_TRANSLATOR_IN_MAIN_MODULE = _T("NEL_SE_TRANS");
|
|
TCHAR envBuf [2];
|
|
if ( GetEnvironmentVariable( SE_TRANSLATOR_IN_MAIN_MODULE, envBuf, 2 ) != 0)
|
|
{
|
|
SetEnvironmentVariable( SE_TRANSLATOR_IN_MAIN_MODULE, NULL );
|
|
}
|
|
|
|
string arg = " " + arguments;
|
|
BOOL res = CreateProcessA(programName.c_str(), (char*)arg.c_str(), NULL, NULL, FALSE, CREATE_DEFAULT_ERROR_MODE | CREATE_NO_WINDOW, NULL, NULL, &si, &pi);
|
|
|
|
if (res)
|
|
{
|
|
//nldebug("LAUNCH: Successful launch '%s' with arg '%s'", programName.c_str(), arguments.c_str());
|
|
CloseHandle( pi.hProcess );
|
|
CloseHandle( pi.hThread );
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
LPVOID lpMsgBuf;
|
|
FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMsgBuf, 0, NULL);
|
|
if (log)
|
|
nlwarning("LAUNCH: Failed launched '%s' with arg '%s' err %d: '%s'", programName.c_str(), arguments.c_str(), GetLastError(), lpMsgBuf);
|
|
LocalFree(lpMsgBuf);
|
|
CloseHandle( pi.hProcess );
|
|
CloseHandle( pi.hThread );
|
|
}
|
|
|
|
#elif defined(NL_OS_MAC)
|
|
std::string command;
|
|
|
|
if (CFile::getExtension(programName) == "app")
|
|
{
|
|
// we need to open bundles with "open" command
|
|
command = NLMISC::toString("open \"%s\"", programName.c_str());
|
|
|
|
// append arguments if any
|
|
if (!arguments.empty())
|
|
{
|
|
command += NLMISC::toString(" --args %s", arguments.c_str());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
command = programName;
|
|
|
|
// append arguments if any
|
|
if (!arguments.empty()) command += " " + arguments;
|
|
}
|
|
|
|
int res = system(command.c_str());
|
|
|
|
if (res && log)
|
|
nlwarning ("LAUNCH: Failed launched '%s' with arg '%s' return code %d", programName.c_str(), arguments.c_str(), res);
|
|
#else
|
|
|
|
static bool firstLaunchProgram = true;
|
|
if (firstLaunchProgram)
|
|
{
|
|
// The aim of this is to avoid defunct process.
|
|
//
|
|
// From "man signal":
|
|
//------
|
|
// According to POSIX (3.3.1.3) it is unspecified what happens when SIGCHLD is set to SIG_IGN. Here
|
|
// the BSD and SYSV behaviours differ, causing BSD software that sets the action for SIGCHLD to
|
|
// SIG_IGN to fail on Linux.
|
|
//------
|
|
//
|
|
// But it works fine on my GNU/Linux so I do this because it's easier :) and I don't know exactly
|
|
// what to do to be portable.
|
|
signal(SIGCHLD, SIG_IGN);
|
|
|
|
firstLaunchProgram = false;
|
|
}
|
|
|
|
// convert one arg into several args
|
|
vector<string> args;
|
|
string::size_type pos1 = 0, pos2 = 0;
|
|
do
|
|
{
|
|
pos1 = arguments.find_first_not_of (" ", pos2);
|
|
if (pos1 == string::npos) break;
|
|
pos2 = arguments.find_first_of (" ", pos1);
|
|
args.push_back (arguments.substr (pos1, pos2-pos1));
|
|
}
|
|
while (pos2 != string::npos);
|
|
|
|
// Store the size of each arg
|
|
vector<char *> argv(args.size()+2);
|
|
uint i = 0;
|
|
argv[i] = (char *)programName.c_str();
|
|
for (; i < args.size(); i++)
|
|
{
|
|
argv[i+1] = (char *) args[i].c_str();
|
|
}
|
|
argv[i+1] = NULL;
|
|
|
|
int status = vfork ();
|
|
/////////////////////////////////////////////////////////
|
|
/// WARNING : NO MORE INSTRUCTION AFTER VFORK !
|
|
/// READ VFORK manual
|
|
/////////////////////////////////////////////////////////
|
|
if (status == -1)
|
|
{
|
|
char *err = strerror (errno);
|
|
if (log)
|
|
nlwarning("LAUNCH: Failed launched '%s' with arg '%s' err %d: '%s'", programName.c_str(), arguments.c_str(), errno, err);
|
|
}
|
|
else if (status == 0)
|
|
{
|
|
// Exec (the only allowed instruction after vfork)
|
|
status = execvp(programName.c_str(), &argv.front());
|
|
|
|
if (status == -1)
|
|
{
|
|
perror("Failed launched");
|
|
_exit(EXIT_FAILURE);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
//nldebug("LAUNCH: Successful launch '%s' with arg '%s'", programName.c_str(), arguments.c_str());
|
|
|
|
return true;
|
|
}
|
|
#endif
|
|
|
|
return false;
|
|
}
|
|
|
|
sint launchProgramAndWaitForResult(const std::string &programName, const std::string &arguments, bool log)
|
|
{
|
|
sint res = 0;
|
|
|
|
#ifdef NL_OS_WINDOWS
|
|
STARTUPINFOA si;
|
|
PROCESS_INFORMATION pi;
|
|
|
|
memset(&si, 0, sizeof(si));
|
|
memset(&pi, 0, sizeof(pi));
|
|
|
|
si.cb = sizeof(si);
|
|
|
|
// Enable nlassert/nlstop to display the error reason & callstack
|
|
const TCHAR *SE_TRANSLATOR_IN_MAIN_MODULE = _T("NEL_SE_TRANS");
|
|
TCHAR envBuf [2];
|
|
if ( GetEnvironmentVariable( SE_TRANSLATOR_IN_MAIN_MODULE, envBuf, 2 ) != 0)
|
|
{
|
|
SetEnvironmentVariable( SE_TRANSLATOR_IN_MAIN_MODULE, NULL );
|
|
}
|
|
|
|
string arg = " " + arguments;
|
|
BOOL ok = CreateProcessA(programName.c_str(), (char*)arg.c_str(), NULL, NULL, FALSE, CREATE_DEFAULT_ERROR_MODE | CREATE_NO_WINDOW, NULL, NULL, &si, &pi);
|
|
|
|
if (ok)
|
|
{
|
|
// Successfully created the process. Wait for it to finish.
|
|
WaitForSingleObject(pi.hProcess, INFINITE);
|
|
|
|
// Get the exit code.
|
|
DWORD exitCode = 0;
|
|
ok = GetExitCodeProcess(pi.hProcess, &exitCode);
|
|
|
|
//nldebug("LAUNCH: Successful launch '%s' with arg '%s'", programName.c_str(), arguments.c_str());
|
|
CloseHandle(pi.hProcess);
|
|
CloseHandle(pi.hThread);
|
|
|
|
if (ok)
|
|
{
|
|
res = (sint)exitCode;
|
|
}
|
|
else
|
|
{
|
|
if (log)
|
|
nlwarning("LAUNCH: Failed launched '%s' with arg '%s'", programName.c_str(), arguments.c_str());
|
|
}
|
|
}
|
|
else
|
|
{
|
|
LPVOID lpMsgBuf;
|
|
FormatMessage(FORMAT_MESSAGE_ALLOCATE_BUFFER | FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS, NULL, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT), (LPTSTR) &lpMsgBuf, 0, NULL);
|
|
|
|
if (log)
|
|
nlwarning("LAUNCH: Failed launched '%s' with arg '%s' err %d: '%s'", programName.c_str(), arguments.c_str(), GetLastError(), lpMsgBuf);
|
|
|
|
LocalFree(lpMsgBuf);
|
|
|
|
CloseHandle(pi.hProcess);
|
|
CloseHandle(pi.hThread);
|
|
}
|
|
#else
|
|
// program name is the only required string
|
|
std::string command = programName;
|
|
|
|
// only appends arguments if any
|
|
if (!arguments.empty()) command += " " + arguments;
|
|
|
|
// execute the command
|
|
res = system(command.c_str());
|
|
|
|
if (res && log)
|
|
nlwarning ("LAUNCH: Failed launched '%s' with arg '%s' return code %d", programName.c_str(), arguments.c_str(), res);
|
|
#endif
|
|
|
|
return res;
|
|
}
|
|
|
|
std::string getCommandOutput(const std::string &command)
|
|
{
|
|
FILE *pipe = popen(command.c_str(), "r");
|
|
|
|
if (!pipe) return "";
|
|
|
|
char buffer[MAX_LINE_WIDTH];
|
|
std::string result;
|
|
|
|
while (!feof(pipe))
|
|
{
|
|
if (fgets(buffer, MAX_LINE_WIDTH, pipe) != NULL) result += buffer;
|
|
}
|
|
|
|
pclose(pipe);
|
|
|
|
return result;
|
|
}
|
|
|
|
std::string expandEnvironmentVariables(const std::string &s)
|
|
{
|
|
size_t len = s.length();
|
|
std::string ret;
|
|
|
|
std::string::size_type pos1 = 0, pos2 = 0;
|
|
|
|
// look for environement variables delimiters
|
|
while(pos2 < len && (pos1 = s.find_first_of("%$", pos2)) != std::string::npos)
|
|
{
|
|
// copy string unprocessed part
|
|
ret += s.substr(pos2, pos1-pos2);
|
|
|
|
// extract a valid variable name (a-zA-Z0-9_)
|
|
pos2 = pos1+1;
|
|
|
|
while(pos2 < len && (isalnum(s[pos2]) || s[pos2] == '_')) ++pos2;
|
|
|
|
// check if variable name is empty
|
|
bool found = pos2 > pos1+1;
|
|
|
|
std::string name;
|
|
|
|
if (found)
|
|
{
|
|
// found at least 1 character
|
|
name = s.substr(pos1+1, pos2-pos1-1);
|
|
}
|
|
|
|
// Windows format needs a trailing % delimiter
|
|
if (found && s[pos1] == '%')
|
|
{
|
|
if (pos2 >= len || s[pos2] != '%')
|
|
{
|
|
// not a variable name, because no trailing %
|
|
found = false;
|
|
}
|
|
else
|
|
{
|
|
// found a trailing %, next character to check
|
|
++pos2;
|
|
}
|
|
}
|
|
|
|
// get variable value if name found
|
|
if (found)
|
|
{
|
|
const char *value = getenv(name.c_str());
|
|
|
|
if (value)
|
|
{
|
|
// value found
|
|
ret += std::string(value);
|
|
}
|
|
else
|
|
{
|
|
// value not found
|
|
found = false;
|
|
}
|
|
}
|
|
|
|
if (!found)
|
|
{
|
|
// variable or value not found, don't evaluate variable
|
|
ret += s.substr(pos1, pos2-pos1);
|
|
}
|
|
}
|
|
|
|
// copy last unprocessed part
|
|
ret += s.substr(pos2);
|
|
|
|
return ret;
|
|
}
|
|
|
|
/*
|
|
* Display the bits (with 0 and 1) composing a byte (from right to left)
|
|
*/
|
|
void displayByteBits( uint8 b, uint nbits, sint beginpos, bool displayBegin, NLMISC::CLog *log )
|
|
{
|
|
string s1, s2;
|
|
sint i;
|
|
for ( i=nbits-1; i!=-1; --i )
|
|
{
|
|
s1 += ( (b >> i) & 1 ) ? '1' : '0';
|
|
}
|
|
log->displayRawNL( "%s", s1.c_str() );
|
|
if ( displayBegin )
|
|
{
|
|
for ( i=nbits; i>beginpos+1; --i )
|
|
{
|
|
s2 += " ";
|
|
}
|
|
s2 += "^";
|
|
log->displayRawNL( "%s beginpos=%u", s2.c_str(), beginpos );
|
|
}
|
|
}
|
|
|
|
|
|
//#define displayDwordBits(a,b,c)
|
|
|
|
/*
|
|
* Display the bits (with 0 and 1) composing a number (uint32) (from right to left)
|
|
*/
|
|
void displayDwordBits( uint32 b, uint nbits, sint beginpos, bool displayBegin, NLMISC::CLog *log )
|
|
{
|
|
string s1, s2;
|
|
sint i;
|
|
for ( i=nbits-1; i!=-1; --i )
|
|
{
|
|
s1 += ( (b >> i) & 1 ) ? '1' : '0';
|
|
}
|
|
log->displayRawNL( "%s", s1.c_str() );
|
|
if ( displayBegin )
|
|
{
|
|
for ( i=nbits; i>beginpos+1; --i )
|
|
{
|
|
s2 += " ";
|
|
}
|
|
s2 += "^";
|
|
log->displayRawNL( "%s beginpos=%u", s2.c_str(), beginpos );
|
|
}
|
|
}
|
|
|
|
|
|
int nlfseek64( FILE *stream, sint64 offset, int origin )
|
|
{
|
|
#ifdef NL_OS_WINDOWS
|
|
|
|
//
|
|
fpos_t pos64 = 0;
|
|
switch (origin)
|
|
{
|
|
case SEEK_CUR:
|
|
if (fgetpos(stream, &pos64) != 0)
|
|
return -1;
|
|
case SEEK_END:
|
|
pos64 = _filelengthi64(_fileno(stream));
|
|
if (pos64 == -1L)
|
|
return -1;
|
|
};
|
|
|
|
// Seek
|
|
pos64 += offset;
|
|
|
|
// Set the final position
|
|
return fsetpos (stream, &pos64);
|
|
|
|
#else // NL_OS_WINDOWS
|
|
// TODO: to fix for Linux and Mac OS X
|
|
|
|
// This code doesn't work under windows : fseek() implementation uses a signed 32 bits offset. What ever we do, it can't seek more than 2 Go.
|
|
// For the moment, i don't know if it works under linux for seek of more than 2 Go.
|
|
|
|
nlassert ((offset < SINT64_CONSTANT(2147483647)) && (offset > SINT64_CONSTANT(-2147483648)));
|
|
|
|
bool first = true;
|
|
do
|
|
{
|
|
// Get the size of the next fseek
|
|
sint nextSeek;
|
|
if (offset > 0)
|
|
nextSeek = (sint)std::min ((sint64)SINT64_CONSTANT(2147483647), offset);
|
|
else
|
|
nextSeek = (sint)std::max ((sint64)-SINT64_CONSTANT(2147483648), offset);
|
|
|
|
// Make a seek
|
|
int result = fseek ( stream, nextSeek, first?origin:SEEK_CUR );
|
|
if (result != 0)
|
|
return result;
|
|
|
|
// Remaining
|
|
offset -= nextSeek;
|
|
first = false;
|
|
}
|
|
while (offset);
|
|
|
|
return 0;
|
|
|
|
#endif // NL_OS_WINDOWS
|
|
}
|
|
|
|
sint64 nlftell64(FILE *stream)
|
|
{
|
|
#ifdef NL_OS_WINDOWS
|
|
fpos_t pos64 = 0;
|
|
if (fgetpos(stream, &pos64) == 0)
|
|
{
|
|
return (sint64) pos64;
|
|
}
|
|
else return -1;
|
|
#else
|
|
nlunreferenced(stream);
|
|
|
|
// TODO: implement for Linux and Mac OS X
|
|
nlerror("Not implemented");
|
|
return -1;
|
|
#endif
|
|
}
|
|
|
|
|
|
//////////////////////////////////////////////////////////////////////////
|
|
//////////////////////////////////////////////////////////////////////////
|
|
/// Commands
|
|
//////////////////////////////////////////////////////////////////////////
|
|
//////////////////////////////////////////////////////////////////////////
|
|
|
|
NLMISC_CATEGORISED_COMMAND(nel, sleep, "Freeze the service for N seconds (for debug purpose)", "<N>")
|
|
{
|
|
nlunreferenced(rawCommandString);
|
|
nlunreferenced(quiet);
|
|
nlunreferenced(human);
|
|
|
|
if(args.size() != 1) return false;
|
|
|
|
sint32 n;
|
|
fromString(args[0], n);
|
|
|
|
log.displayNL ("Sleeping during %d seconds", n);
|
|
|
|
nlSleep(n * 1000);
|
|
return true;
|
|
}
|
|
|
|
NLMISC_CATEGORISED_COMMAND(nel, system, "Execute the command line using system() function call (wait until the end of the command)", "<commandline>")
|
|
{
|
|
nlunreferenced(rawCommandString);
|
|
nlunreferenced(quiet);
|
|
nlunreferenced(human);
|
|
|
|
if(args.size() != 1) return false;
|
|
|
|
string cmd = args[0];
|
|
log.displayNL ("Executing '%s'", cmd.c_str());
|
|
sint error = system(cmd.c_str());
|
|
if (error)
|
|
{
|
|
log.displayNL ("Execution of '%s' failed with error code %d", cmd.c_str(), error);
|
|
}
|
|
else
|
|
{
|
|
log.displayNL ("End of Execution of '%s'", cmd.c_str());
|
|
}
|
|
return true;
|
|
}
|
|
|
|
NLMISC_CATEGORISED_COMMAND(nel, launchProgram, "Execute the command line using launcProgram() function call (launch in background task without waiting the end of the execution)", "<programName> <arguments>")
|
|
{
|
|
nlunreferenced(rawCommandString);
|
|
nlunreferenced(quiet);
|
|
nlunreferenced(human);
|
|
|
|
if(args.size() != 2) return false;
|
|
|
|
string cmd = args[0];
|
|
string arg = args[1];
|
|
log.displayNL ("Executing '%s' with argument '%s'", cmd.c_str(), arg.c_str());
|
|
launchProgram(cmd, arg);
|
|
log.displayNL ("End of Execution of '%s' with argument '%s'", cmd.c_str(), arg.c_str());
|
|
return true;
|
|
}
|
|
|
|
NLMISC_CATEGORISED_COMMAND(nel, killProgram, "kill a program given the pid", "<pid>")
|
|
{
|
|
nlunreferenced(rawCommandString);
|
|
nlunreferenced(quiet);
|
|
nlunreferenced(human);
|
|
|
|
if(args.size() != 1) return false;
|
|
uint32 pid;
|
|
fromString(args[0], pid);
|
|
killProgram(pid);
|
|
return true;
|
|
}
|
|
|
|
#ifdef NL_OS_WINDOWS
|
|
LONG GetRegKey(HKEY key, LPCSTR subkey, LPSTR retdata)
|
|
{
|
|
HKEY hkey;
|
|
LONG retval = RegOpenKeyExA(key, subkey, 0, KEY_QUERY_VALUE, &hkey);
|
|
|
|
if (retval == ERROR_SUCCESS)
|
|
{
|
|
long datasize = MAX_PATH;
|
|
char data[MAX_PATH];
|
|
RegQueryValueA(hkey, NULL, data, &datasize);
|
|
lstrcpyA(retdata,data);
|
|
RegCloseKey(hkey);
|
|
}
|
|
|
|
return retval;
|
|
}
|
|
#endif // NL_OS_WINDOWS
|
|
|
|
static bool openDocWithExtension (const char *document, const char *ext)
|
|
{
|
|
#ifdef NL_OS_WINDOWS
|
|
// First try ShellExecute()
|
|
HINSTANCE result = ShellExecuteA(NULL, "open", document, NULL, NULL, SW_SHOWDEFAULT);
|
|
|
|
// If it failed, get the .htm regkey and lookup the program
|
|
if ((uintptr_t)result <= HINSTANCE_ERROR)
|
|
{
|
|
char key[MAX_PATH + MAX_PATH];
|
|
|
|
if (GetRegKey(HKEY_CLASSES_ROOT, ext, key) == ERROR_SUCCESS)
|
|
{
|
|
lstrcatA(key, "\\shell\\open\\command");
|
|
|
|
if (GetRegKey(HKEY_CLASSES_ROOT, key, key) == ERROR_SUCCESS)
|
|
{
|
|
char *pos = strstr(key, "\"%1\"");
|
|
|
|
if (pos == NULL)
|
|
{
|
|
// No quotes found
|
|
// Check for %1, without quotes
|
|
pos = strstr(key, "%1");
|
|
|
|
if (pos == NULL)
|
|
{
|
|
// No parameter at all...
|
|
pos = key+lstrlenA(key)-1;
|
|
}
|
|
else
|
|
{
|
|
// Remove the parameter
|
|
*pos = '\0';
|
|
}
|
|
}
|
|
else
|
|
{
|
|
// Remove the parameter
|
|
*pos = '\0';
|
|
}
|
|
|
|
lstrcatA(pos, " ");
|
|
lstrcatA(pos, document);
|
|
int res = WinExec(key, SW_SHOWDEFAULT);
|
|
return (res>31);
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
return true;
|
|
}
|
|
#elif defined(NL_OS_MAC)
|
|
CFURLRef url = CFURLCreateWithBytes(NULL, (const UInt8 *)document, strlen(document), kCFStringEncodingUTF8, NULL);
|
|
|
|
if (url)
|
|
{
|
|
OSStatus res = LSOpenCFURLRef(url, 0);
|
|
CFRelease(url);
|
|
|
|
if (res != 0)
|
|
{
|
|
nlwarning("LSOpenCFURLRef %s returned %d", document, (sint)res);
|
|
return false;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
else
|
|
{
|
|
nlwarning("Unable to create URL from %s", document);
|
|
return false;
|
|
}
|
|
#else
|
|
std::string command = "/usr/bin/xdg-open";
|
|
|
|
if (!CFile::fileExists(command))
|
|
{
|
|
if (strcmp(ext, "htm") == 0)
|
|
{
|
|
command = "/etc/alternatives/x-www-browser";
|
|
|
|
if (!CFile::fileExists(command))
|
|
{
|
|
command.clear();
|
|
}
|
|
}
|
|
else
|
|
{
|
|
command.clear();
|
|
}
|
|
}
|
|
|
|
if (command.empty())
|
|
{
|
|
nlwarning("Unable to open %s", document);
|
|
return false;
|
|
}
|
|
|
|
// save LD_LIBRARY_PATH
|
|
const char *previousEnv = getenv("LD_LIBRARY_PATH");
|
|
|
|
// clear LD_LIBRARY_PATH to avoid problems with Steam Runtime
|
|
setenv("LD_LIBRARY_PATH", "", 1);
|
|
|
|
bool res = launchProgram(command, document);
|
|
|
|
// restore previous LD_LIBRARY_PATH
|
|
setenv("LD_LIBRARY_PATH", previousEnv, 1);
|
|
|
|
return res;
|
|
#endif // NL_OS_WINDOWS
|
|
|
|
return false;
|
|
}
|
|
|
|
bool openURL (const char *url)
|
|
{
|
|
return openDocWithExtension(url, "htm");
|
|
}
|
|
|
|
bool openDoc (const char *document)
|
|
{
|
|
// get extension from document fullpath
|
|
string ext = CFile::getExtension(document);
|
|
|
|
// try to open document
|
|
return openDocWithExtension(document, ext.c_str());
|
|
}
|
|
|
|
} // NLMISC
|