2012-05-29 13:31:11 +00:00
// 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"
2013-02-08 12:29:17 +00:00
# include "nel/misc/types_nl.h"
# include "nel/misc/common.h"
2012-05-29 13:31:11 +00:00
# ifdef NL_OS_WINDOWS
2015-02-24 11:47:54 +00:00
# include <ShellAPI.h>
2012-05-29 13:31:11 +00:00
# include <io.h>
# include <tchar.h>
2015-12-25 14:29:23 +00:00
# define popen _popen
# define pclose _pclose
2015-11-24 09:02:09 +00:00
# elif defined NL_OS_MAC
# include <ApplicationServices / ApplicationServices.h>
2012-05-29 13:31:11 +00:00
# elif defined NL_OS_UNIX
# include <unistd.h>
# include <cerrno>
# include <pthread.h>
# include <sched.h>
# endif
2015-12-25 14:29:23 +00:00
# define MAX_LINE_WIDTH 256
2012-05-29 13:31:11 +00:00
# include "nel/misc/command.h"
# include "nel/misc/path.h"
# include "nel/misc/i18n.h"
using namespace std ;
2014-06-17 19:48:25 +00:00
# ifndef NL_COMP_MINGW
2012-05-29 13:31:11 +00:00
# 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
2014-06-17 19:48:25 +00:00
# endif // !NL_COMP_MINGW
2012-05-29 13:31:11 +00:00
2016-01-04 12:48:57 +00:00
# if defined(NL_HAS_SSE2) && !defined(NL_CPU_X86_64)
2014-06-13 17:12:31 +00:00
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 */
2013-02-08 12:29:17 +00:00
# ifdef DEBUG_NEW
# define new DEBUG_NEW
# endif
2012-05-29 13:31:11 +00:00
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 )
{
2015-12-23 14:11:26 +00:00
static const char * divTable [ ] = { " B " , " KiB " , " MiB " , " GiB " , " TiB " } ;
2012-05-29 13:31:11 +00:00
uint div = 0 ;
uint64 res = bytes ;
uint64 newres = res ;
for ( ; ; )
{
newres / = 1024 ;
if ( newres < 8 | | div > 3 )
break ;
div + + ;
res = newres ;
}
2016-02-02 11:37:06 +00:00
return toString ( " % " NL_I64 " u %s " , res , divTable [ div ] ) ;
}
std : : string bytesToHumanReadableUnits ( uint64 bytes , const std : : vector < std : : string > & units )
{
if ( units . empty ( ) ) return " " ;
uint div = 0 ;
uint last = units . size ( ) - 1 ;
uint64 res = bytes ;
uint64 newres = res ;
for ( ; ; )
{
newres / = 1024 ;
if ( newres < 8 | | div > 3 | | div = = last )
break ;
+ + div ;
res = newres ;
}
return toString ( " % " NL_I64 " u %s " , res , units [ div ] . c_str ( ) ) ;
2012-05-29 13:31:11 +00:00
}
uint32 humanReadableToBytes ( const string & str )
{
uint32 res ;
if ( str . empty ( ) )
return 0 ;
// not a number
if ( str [ 0 ] < ' 0 ' | | str [ 0 ] > ' 9 ' )
return 0 ;
2016-02-02 10:33:44 +00:00
if ( ! fromString ( str , res ) )
return 0 ;
2012-05-29 13:31:11 +00:00
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 ] )
{
2016-02-02 10:33:44 +00:00
// kB/KB, MB, GB and TB are 1000 multiples
case ' T ' : res * = 1000 ;
case ' G ' : res * = 1000 ;
case ' M ' : res * = 1000 ;
case ' k ' : res * = 1000 ; break ; // kilo symbol should be a lowercase K
case ' K ' : res * = 1000 ; break ;
case ' i ' :
{
// KiB, MiB, GiB and TiB are 1024 multiples
if ( str . size ( ) < 4 )
return res ;
switch ( str [ str . size ( ) - 3 ] )
{
case ' T ' : res * = 1024 ;
case ' G ' : res * = 1024 ;
case ' M ' : res * = 1024 ;
case ' K ' : res * = 1024 ;
default : ;
}
}
default : ;
2012-05-29 13:31:11 +00:00
}
}
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 " } ;
2015-11-07 17:33:16 +00:00
static uint divCoef [ ] = { 60 , 60 , 24 } ;
2012-05-29 13:31:11 +00:00
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 ] ) ;
}
2016-02-20 17:10:17 +00:00
std : : string timestampToHumanReadable ( uint32 timestamp )
{
char buffer [ 30 ] ;
time_t dtime = timestamp ;
tm * tms = localtime ( & dtime ) ;
if ( tms )
{
strftime ( buffer , 30 , " %Y-%m-%d %H:%M:%S " , tms ) ;
return std : : string ( buffer ) ;
}
return " " ;
}
2012-05-29 13:31:11 +00:00
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 ) ;
nlwarning ( " Failed to kill '%d' err %d: '%s' " , pid , GetLastError ( ) , 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
}
# ifdef NL_OS_WINDOWS
2016-02-20 17:54:32 +00:00
static bool createProcess ( const std : : string & programName , const std : : string & arguments , bool log , PROCESS_INFORMATION & pi )
{
STARTUPINFOW si ;
2015-11-07 17:33:16 +00:00
memset ( & si , 0 , sizeof ( si ) ) ;
memset ( & pi , 0 , sizeof ( pi ) ) ;
2012-05-29 13:31:11 +00:00
2015-11-07 17:33:16 +00:00
si . cb = sizeof ( si ) ;
2012-05-29 13:31:11 +00:00
// Enable nlassert/nlstop to display the error reason & callstack
2016-02-20 17:54:32 +00:00
const char * SE_TRANSLATOR_IN_MAIN_MODULE = " NEL_SE_TRANS " ;
char envBuf [ 2 ] ;
if ( GetEnvironmentVariableA ( SE_TRANSLATOR_IN_MAIN_MODULE , envBuf , 2 ) ! = 0 )
2012-05-29 13:31:11 +00:00
{
2016-02-20 17:54:32 +00:00
SetEnvironmentVariableA ( SE_TRANSLATOR_IN_MAIN_MODULE , NULL ) ;
2012-05-29 13:31:11 +00:00
}
2016-02-20 17:54:32 +00:00
wchar_t * sProgramName = NULL ;
2016-02-02 18:49:37 +00:00
std : : string args ;
// a .bat file must have first parameter to NULL and use 2nd parameter to pass filename
if ( CFile : : getExtension ( programName ) = = " bat " )
{
args = " \" " + programName + " \" " + arguments ;
}
else
{
2016-02-20 17:54:32 +00:00
ucstring ucProgramName ;
ucProgramName . fromUtf8 ( programName ) ;
sProgramName = new wchar_t [ MAX_PATH ] ;
wcscpy ( sProgramName , ( wchar_t * ) ucProgramName . c_str ( ) ) ;
2016-12-02 15:00:22 +00:00
// important! we need to specify the executable full path as first argument
args = toString ( " \" %s \" " , programName . c_str ( ) ) + arguments ;
2016-02-02 18:49:37 +00:00
}
2016-02-20 17:54:32 +00:00
BOOL res = CreateProcessW ( sProgramName , utf8ToWide ( args ) , NULL , NULL , FALSE , CREATE_DEFAULT_ERROR_MODE | CREATE_NO_WINDOW , NULL , NULL , & si , & pi ) ;
2012-05-29 13:31:11 +00:00
2016-02-20 17:54:32 +00:00
if ( sProgramName )
2012-05-29 13:31:11 +00:00
{
2016-02-20 17:54:32 +00:00
delete [ ] sProgramName ;
sProgramName = NULL ;
2012-05-29 13:31:11 +00:00
}
2016-02-20 17:54:32 +00:00
if ( ! res )
2012-05-29 13:31:11 +00:00
{
2015-03-06 20:07:12 +00:00
if ( log )
2016-02-12 15:04:18 +00:00
{
sint lastError = getLastError ( ) ;
nlwarning ( " LAUNCH: Failed launched '%s' with arg '%s' err %d: '%s' " , programName . c_str ( ) , arguments . c_str ( ) , lastError , formatErrorMessage ( lastError ) . c_str ( ) ) ;
}
2012-05-29 13:31:11 +00:00
CloseHandle ( pi . hProcess ) ;
CloseHandle ( pi . hThread ) ;
2016-02-20 17:54:32 +00:00
return false ;
2012-05-29 13:31:11 +00:00
}
2016-02-20 17:54:32 +00:00
return true ;
}
# endif
bool launchProgram ( const std : : string & programName , const std : : string & arguments , bool log )
{
# ifdef NL_OS_WINDOWS
PROCESS_INFORMATION pi ;
if ( ! createProcess ( programName , arguments , log , pi ) ) return false ;
//nldebug("LAUNCH: Successful launch '%s' with arg '%s'", programName.c_str(), arguments.c_str());
CloseHandle ( pi . hProcess ) ;
CloseHandle ( pi . hThread ) ;
return true ;
2016-02-13 22:30:00 +00:00
# else
2015-12-15 13:06:49 +00:00
2016-02-13 22:30:00 +00:00
# ifdef NL_OS_MAC
// special OS X case with bundles
2016-02-28 15:44:08 +00:00
if ( toLower ( CFile : : getExtension ( programName ) ) = = " .app " )
2015-12-15 13:06:49 +00:00
{
2016-02-13 22:30:00 +00:00
// we need to open bundles with "open" command
std : : string command = NLMISC : : toString ( " open \" %s \" " , programName . c_str ( ) ) ;
2015-12-15 13:06:49 +00:00
2016-02-13 22:30:00 +00:00
// append arguments if any
if ( ! arguments . empty ( ) )
{
command + = NLMISC : : toString ( " --args %s " , arguments . c_str ( ) ) ;
}
2015-12-15 13:06:49 +00:00
2016-02-13 22:30:00 +00:00
int res = system ( command . c_str ( ) ) ;
2016-02-13 12:57:43 +00:00
2016-02-13 22:30:00 +00:00
if ( ! res ) return true ;
if ( log )
{
nlwarning ( " LAUNCH: Failed launched '%s' with arg '%s' return code %d " , programName . c_str ( ) , arguments . c_str ( ) , res ) ;
}
return false ;
2016-02-13 12:57:43 +00:00
}
2016-02-13 22:30:00 +00:00
# endif
2012-05-29 13:31:11 +00:00
static bool firstLaunchProgram = true ;
2016-02-27 17:38:58 +00:00
2012-05-29 13:31:11 +00:00
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.
2015-11-21 16:37:45 +00:00
signal ( SIGCHLD , SIG_IGN ) ;
2012-05-29 13:31:11 +00:00
firstLaunchProgram = false ;
}
// convert one arg into several args
vector < string > args ;
2016-02-27 17:37:43 +00:00
explodeArguments ( arguments , args ) ;
2012-05-29 13:31:11 +00:00
// 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 ;
2015-11-21 16:39:49 +00:00
2016-02-27 17:38:58 +00:00
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 ;
}
bool launchProgramArray ( const std : : string & programName , const std : : vector < std : : string > & arguments , bool log )
{
# ifdef NL_OS_WINDOWS
PROCESS_INFORMATION pi ;
std : : string argumentsJoined = joinArguments ( arguments ) ;
if ( ! createProcess ( programName , argumentsJoined , log , pi ) ) return false ;
//nldebug("LAUNCH: Successful launch '%s' with arg '%s'", programName.c_str(), arguments.c_str());
CloseHandle ( pi . hProcess ) ;
CloseHandle ( pi . hThread ) ;
return true ;
# else
# ifdef NL_OS_MAC
// special OS X case with bundles
2016-02-28 15:44:08 +00:00
if ( toLower ( CFile : : getExtension ( programName ) ) = = " app " )
2016-02-27 17:38:58 +00:00
{
// we need to open bundles with "open" command
std : : string command = NLMISC : : toString ( " open \" %s \" " , programName . c_str ( ) ) ;
2016-02-28 11:45:31 +00:00
std : : string argumentsJoined = joinArguments ( arguments ) ;
2016-02-27 17:38:58 +00:00
// append arguments if any
2016-02-28 11:45:31 +00:00
if ( ! argumentsJoined . empty ( ) )
2016-02-27 17:38:58 +00:00
{
2016-02-28 11:45:31 +00:00
command + = NLMISC : : toString ( " --args %s " , argumentsJoined . c_str ( ) ) ;
2016-02-27 17:38:58 +00:00
}
int res = system ( command . c_str ( ) ) ;
if ( ! res ) return true ;
if ( log )
{
2016-02-28 11:45:31 +00:00
nlwarning ( " LAUNCH: Failed launched '%s' with arg '%s' return code %d " , programName . c_str ( ) , argumentsJoined . c_str ( ) , res ) ;
2016-02-27 17:38:58 +00:00
}
return false ;
}
# endif
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 ;
}
// Store the size of each arg
vector < char * > argv ( arguments . size ( ) + 2 ) ;
uint i = 0 ;
argv [ i ] = ( char * ) programName . c_str ( ) ;
for ( ; i < arguments . size ( ) ; i + + )
{
argv [ i + 1 ] = ( char * ) arguments [ i ] . c_str ( ) ;
}
argv [ i + 1 ] = NULL ;
2012-05-29 13:31:11 +00:00
int status = vfork ( ) ;
/////////////////////////////////////////////////////////
2015-11-21 16:37:45 +00:00
/// WARNING : NO MORE INSTRUCTION AFTER VFORK !
2012-05-29 13:31:11 +00:00
/// READ VFORK manual
/////////////////////////////////////////////////////////
if ( status = = - 1 )
{
char * err = strerror ( errno ) ;
2015-03-06 20:07:12 +00:00
if ( log )
2016-02-27 18:22:23 +00:00
nlwarning ( " LAUNCH: Failed launched '%s' with arg '%s' err %d: '%s' " , programName . c_str ( ) , joinArguments ( arguments ) . c_str ( ) , errno , err ) ;
2012-05-29 13:31:11 +00:00
}
else if ( status = = 0 )
2015-11-07 17:33:16 +00:00
{
2012-05-29 13:31:11 +00:00
// 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());
2015-11-21 16:39:49 +00:00
2012-05-29 13:31:11 +00:00
return true ;
}
# endif
return false ;
2016-01-02 18:16:33 +00:00
}
2012-05-29 13:31:11 +00:00
2016-01-02 18:14:13 +00:00
sint launchProgramAndWaitForResult ( const std : : string & programName , const std : : string & arguments , bool log )
{
# ifdef NL_OS_WINDOWS
PROCESS_INFORMATION pi ;
2016-02-20 17:54:32 +00:00
if ( ! createProcess ( programName , arguments , log , pi ) ) return - 1 ;
2016-02-03 20:58:29 +00:00
2016-02-20 17:54:32 +00:00
// Successfully created the process. Wait for it to finish.
WaitForSingleObject ( pi . hProcess , INFINITE ) ;
2016-01-02 18:14:13 +00:00
2016-02-20 17:54:32 +00:00
// Get the exit code.
DWORD exitCode = 0 ;
BOOL ok = GetExitCodeProcess ( pi . hProcess , & exitCode ) ;
2016-01-02 18:14:13 +00:00
2016-02-20 17:54:32 +00:00
//nldebug("LAUNCH: Successful launch '%s' with arg '%s'", programName.c_str(), arguments.c_str());
CloseHandle ( pi . hProcess ) ;
CloseHandle ( pi . hThread ) ;
2016-01-02 18:14:13 +00:00
2016-02-20 17:54:32 +00:00
if ( ok ) return ( sint ) exitCode ;
2016-01-02 18:14:13 +00:00
2016-02-20 17:54:32 +00:00
if ( log )
nlwarning ( " LAUNCH: Failed launched '%s' with arg '%s' " , programName . c_str ( ) , arguments . c_str ( ) ) ;
2016-01-02 18:14:13 +00:00
2016-02-20 17:54:32 +00:00
return - 1 ;
2016-01-02 18:14:13 +00:00
# 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
2016-02-20 17:54:32 +00:00
sint res = system ( command . c_str ( ) ) ;
2016-01-02 18:14:13 +00:00
if ( res & & log )
nlwarning ( " LAUNCH: Failed launched '%s' with arg '%s' return code %d " , programName . c_str ( ) , arguments . c_str ( ) , res ) ;
return res ;
2016-02-20 17:54:32 +00:00
# endif
2012-05-29 13:31:11 +00:00
}
2015-12-25 14:29:23 +00:00
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 ;
}
2016-01-13 18:45:02 +00:00
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 ;
2016-02-13 22:30:18 +00:00
nlwarning ( " Environment variable '%s' not found, won't be replaced " , name . c_str ( ) ) ;
2016-01-13 18:45:02 +00:00
}
}
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 ;
}
2016-02-27 17:37:43 +00:00
bool explodeArguments ( const std : : string & str , std : : vector < std : : string > & args )
{
if ( str . empty ( ) ) return false ;
std : : string : : size_type pos1 = 0 , pos2 = 0 ;
do
{
// Look for the first non space character
pos1 = str . find_first_not_of ( " " , pos2 ) ;
if ( pos1 = = std : : string : : npos ) break ;
// Look for the first space or "
pos2 = str . find_first_of ( " \" " , pos1 ) ;
if ( pos2 ! = std : : string : : npos )
{
// " ?
if ( str [ pos2 ] = = ' " ' )
{
// Look for the final \"
pos2 = str . find_first_of ( " \" " , pos2 + 1 ) ;
if ( pos2 ! = std : : string : : npos )
{
// Look for the first space
pos2 = str . find_first_of ( " " , pos2 + 1 ) ;
}
}
}
// Compute the size of the string to extract
std : : string : : difference_type length = ( pos2 ! = std : : string : : npos ) ? pos2 - pos1 : std : : string : : npos ;
std : : string tmp = str . substr ( pos1 , length ) ;
// remove escape " from argument
if ( tmp . length ( ) > 1 & & tmp [ 0 ] = = ' " ' & & tmp [ tmp . length ( ) - 1 ] = = ' " ' ) tmp = tmp . substr ( 1 , tmp . length ( ) - 2 ) ;
args . push_back ( tmp ) ;
}
while ( pos2 ! = std : : string : : npos ) ;
return true ;
}
std : : string joinArguments ( const std : : vector < std : : string > & args )
{
std : : string res ;
for ( uint i = 0 , len = ( uint ) args . size ( ) ; i < len ; + + i )
{
const std : : string & arg = args [ i ] ;
// prepend space
if ( ! res . empty ( ) ) res + = " " ;
// escape only if spaces or empty argument
if ( arg . empty ( ) | | arg . find ( ' ' ) ! = std : : string : : npos )
{
res + = " \" " + arg + " \" " ;
}
else
{
res + = arg ;
}
}
return res ;
}
2012-05-29 13:31:11 +00:00
/*
* 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 ) ;
}
}
2016-02-20 16:46:11 +00:00
FILE * nlfopen ( const std : : string & filename , const std : : string & mode )
{
# ifdef NL_OS_WINDOWS
return _wfopen ( utf8ToWide ( filename ) , utf8ToWide ( mode ) ) ;
# else
return fopen ( filename . c_str ( ) , mode . c_str ( ) ) ;
# endif
}
2012-05-29 13:31:11 +00:00
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
2016-12-02 14:57:25 +00:00
LONG GetRegKey ( HKEY key , LPCWSTR subkey , LPWSTR retdata )
2012-05-29 13:31:11 +00:00
{
2015-11-07 17:33:16 +00:00
HKEY hkey ;
2016-12-02 14:57:25 +00:00
LONG retval = RegOpenKeyExW ( key , subkey , 0 , KEY_QUERY_VALUE , & hkey ) ;
2012-05-29 13:31:11 +00:00
2015-11-07 17:33:16 +00:00
if ( retval = = ERROR_SUCCESS )
2012-05-29 13:31:11 +00:00
{
2015-11-07 17:33:16 +00:00
long datasize = MAX_PATH ;
2016-12-02 14:57:25 +00:00
wchar_t data [ MAX_PATH ] ;
RegQueryValueW ( hkey , NULL , data , & datasize ) ;
lstrcpyW ( retdata , data ) ;
2015-11-07 17:33:16 +00:00
RegCloseKey ( hkey ) ;
}
return retval ;
2012-05-29 13:31:11 +00:00
}
# endif // NL_OS_WINDOWS
2016-12-02 14:58:59 +00:00
static bool openDocWithExtension ( const std : : string & document , const std : : string & ext )
2012-05-29 13:31:11 +00:00
{
# ifdef NL_OS_WINDOWS
2015-11-07 17:31:58 +00:00
// First try ShellExecute()
2016-12-02 14:57:25 +00:00
HINSTANCE result = ShellExecuteW ( NULL , L " open " , utf8ToWide ( document ) , NULL , NULL , SW_SHOWDEFAULT ) ;
2015-11-07 17:31:58 +00:00
// If it failed, get the .htm regkey and lookup the program
if ( ( uintptr_t ) result < = HINSTANCE_ERROR )
2012-05-29 13:31:11 +00:00
{
2016-12-02 14:57:25 +00:00
wchar_t key [ MAX_PATH + MAX_PATH ] ;
2012-05-29 13:31:11 +00:00
2016-12-02 14:57:25 +00:00
// get the type of the extension
if ( GetRegKey ( HKEY_CLASSES_ROOT , utf8ToWide ( " . " + ext ) , key ) = = ERROR_SUCCESS )
2012-05-29 13:31:11 +00:00
{
2016-12-02 14:57:25 +00:00
lstrcatW ( key , L " \\ shell \\ open \\ command " ) ;
2015-11-07 17:31:58 +00:00
2016-12-02 14:59:45 +00:00
// get the command used to open a file with this extension
2015-11-07 17:31:58 +00:00
if ( GetRegKey ( HKEY_CLASSES_ROOT , key , key ) = = ERROR_SUCCESS )
{
2016-12-02 14:59:45 +00:00
std : : string program = wideToUtf8 ( key ) ;
2015-11-07 17:31:58 +00:00
2016-12-02 14:59:45 +00:00
// empty program
if ( program . empty ( ) ) return false ;
if ( program [ 0 ] = = ' " ' )
2015-11-07 17:31:58 +00:00
{
2016-12-02 14:59:45 +00:00
// program is quoted
std : : string : : size_type pos = program . find ( ' " ' , 1 ) ;
if ( pos ! = std : : string : : npos )
2015-11-07 17:31:58 +00:00
{
2016-12-02 14:59:45 +00:00
// take part before next quote
program = program . substr ( 1 , pos - 1 ) ;
2015-11-07 17:31:58 +00:00
}
}
else
{
2016-12-02 14:59:45 +00:00
// program has a parameter
std : : string : : size_type pos = program . find ( ' ' , 1 ) ;
if ( pos ! = std : : string : : npos )
{
// take part before first space
program = program . substr ( 0 , pos ) ;
}
2015-11-07 17:31:58 +00:00
}
2016-12-02 14:59:45 +00:00
// create process
PROCESS_INFORMATION pi ;
return createProcess ( program , document , false , pi ) ;
2015-11-07 17:31:58 +00:00
}
2012-05-29 13:31:11 +00:00
}
}
2015-11-07 17:31:58 +00:00
else
{
return true ;
}
2015-11-21 16:36:44 +00:00
# elif defined(NL_OS_MAC)
2016-12-05 18:39:19 +00:00
CFURLRef url = CFURLCreateWithBytes ( NULL , ( const UInt8 * ) document . c_str ( ) , document . length ( ) , kCFStringEncodingUTF8 , NULL ) ;
2015-11-24 09:02:09 +00:00
if ( url )
{
OSStatus res = LSOpenCFURLRef ( url , 0 ) ;
CFRelease ( url ) ;
if ( res ! = 0 )
{
2016-12-05 18:39:19 +00:00
nlwarning ( " LSOpenCFURLRef %s returned %d " , document . c_str ( ) , ( sint ) res ) ;
2015-11-24 09:02:09 +00:00
return false ;
}
return true ;
}
else
{
2016-12-05 18:39:19 +00:00
nlwarning ( " Unable to create URL from %s " , document . c_str ( ) ) ;
2015-11-24 09:02:09 +00:00
return false ;
}
2015-11-07 17:31:58 +00:00
# else
2015-11-21 16:36:44 +00:00
std : : string command = " /usr/bin/xdg-open " ;
if ( ! CFile : : fileExists ( command ) )
{
2016-12-05 08:03:13 +00:00
if ( ext = = " htm " )
2015-11-21 16:36:44 +00:00
{
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 ;
}
2016-01-03 16:51:09 +00:00
// save LD_LIBRARY_PATH
const char * previousEnv = getenv ( " LD_LIBRARY_PATH " ) ;
2015-11-21 16:36:44 +00:00
2016-01-03 16:51:09 +00:00
// 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 ;
2015-11-07 17:31:58 +00:00
# endif // NL_OS_WINDOWS
return false ;
}
2016-12-02 14:58:59 +00:00
bool openURL ( const std : : string & url )
2015-11-07 17:31:58 +00:00
{
return openDocWithExtension ( url , " htm " ) ;
2012-05-29 13:31:11 +00:00
}
2016-12-02 14:58:59 +00:00
bool openDoc ( const std : : string & document )
2012-05-29 13:31:11 +00:00
{
2015-11-07 17:31:58 +00:00
// get extension from document fullpath
string ext = CFile : : getExtension ( document ) ;
2012-05-29 13:31:11 +00:00
2015-11-07 17:31:58 +00:00
// try to open document
2016-12-02 14:58:59 +00:00
return openDocWithExtension ( document , ext ) ;
2012-05-29 13:31:11 +00:00
}
} // NLMISC