2833 lines
58 KiB
C
Executable File
2833 lines
58 KiB
C
Executable File
/* Part of SWI-Prolog
|
|
|
|
Author: Jan Wielemaker
|
|
E-mail: J.Wielemaker@vu.nl
|
|
WWW: http://www.swi-prolog.org
|
|
Copyright (C): 1985-2013, University of Amsterdam
|
|
VU University Amsterdam
|
|
|
|
This library is free software; you can redistribute it and/or
|
|
modify 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 (at your option) any later version.
|
|
|
|
This library 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
|
|
Lesser General Public License for more details.
|
|
|
|
You should have received a copy of the GNU Lesser General Public
|
|
License along with this library; if not, write to the Free Software
|
|
Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
*/
|
|
|
|
/* Modified (M) 1993 Dave Sherratt */
|
|
|
|
/*#define O_DEBUG 1*/
|
|
|
|
#if OS2 && EMX
|
|
#include <os2.h> /* this has to appear before pl-incl.h */
|
|
#endif
|
|
|
|
//@{
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
Solaris has asctime_r() with 3 arguments. Using _POSIX_PTHREAD_SEMANTICS
|
|
is supposed to give the POSIX standard one.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
#if defined(__sun__) || defined(__sun)
|
|
#define _POSIX_PTHREAD_SEMANTICS 1
|
|
#endif
|
|
|
|
#define __MINGW_USE_VC2005_COMPAT /* Get Windows time_t as 64-bit */
|
|
|
|
#include "pl-incl.h"
|
|
#include "pl-ctype.h"
|
|
#include "pl-utf8.h"
|
|
#include <stdlib.h>
|
|
#include <math.h>
|
|
#include <stdio.h> /* rename() and remove() prototypes */
|
|
|
|
#if TIME_WITH_SYS_TIME
|
|
# include <sys/time.h>
|
|
# include <time.h>
|
|
#else
|
|
# if HAVE_SYS_TIME_H
|
|
# include <sys/time.h>
|
|
# else
|
|
# include <time.h>
|
|
# endif
|
|
#endif
|
|
|
|
#if HAVE_SYS_STAT_H
|
|
#include <sys/stat.h>
|
|
#endif
|
|
#if HAVE_ALLOCA_H
|
|
#include <alloca.h>
|
|
#endif
|
|
#ifdef O_XOS
|
|
#define statstruct struct _stati64
|
|
#else
|
|
#define statstruct struct stat
|
|
#define statfunc stat
|
|
#endif
|
|
#if HAVE_PWD_H
|
|
#include <pwd.h>
|
|
#endif
|
|
#if HAVE_VFORK_H
|
|
#include <vfork.h>
|
|
#endif
|
|
#ifdef HAVE_UNISTD_H
|
|
#include <unistd.h>
|
|
#endif
|
|
#ifdef HAVE_SYS_FILE_H
|
|
#include <sys/file.h>
|
|
#endif
|
|
#if defined(HAVE_SYS_RESOURCE_H)
|
|
#include <sys/resource.h>
|
|
#endif
|
|
#ifdef HAVE_FTIME
|
|
#include <sys/timeb.h>
|
|
#endif
|
|
#include <time.h>
|
|
#include <fcntl.h>
|
|
#ifndef __WATCOMC__ /* appears a conflict */
|
|
#include <errno.h>
|
|
#endif
|
|
|
|
#if defined(__WATCOMC__)
|
|
#include <io.h>
|
|
#include <dos.h>
|
|
#endif
|
|
|
|
#if OS2 && EMX
|
|
static double initial_time;
|
|
#endif /* OS2 */
|
|
|
|
#define LOCK() PL_LOCK(L_OS)
|
|
#define UNLOCK() PL_UNLOCK(L_OS)
|
|
|
|
static void initExpand(void);
|
|
static void cleanupExpand(void);
|
|
static void initEnviron(void);
|
|
|
|
#ifndef DEFAULT_PATH
|
|
#define DEFAULT_PATH "/bin:/usr/bin"
|
|
#endif
|
|
|
|
/** shell(+Command:text, -Status:integer) is det.
|
|
|
|
Run an external command and wait for its completion.
|
|
*/
|
|
|
|
static
|
|
PRED_IMPL("shell", 2, shell, 0)
|
|
{ GET_LD
|
|
char *cmd;
|
|
|
|
if ( PL_get_chars(A1, &cmd, CVT_ALL|REP_FN|CVT_EXCEPTION) )
|
|
{ int rval = System(cmd);
|
|
|
|
return PL_unify_integer(A2, rval);
|
|
}
|
|
|
|
fail;
|
|
}
|
|
|
|
/********************************
|
|
* INITIALISATION *
|
|
*********************************/
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
bool initOs()
|
|
|
|
Initialise the OS dependant functions.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
bool
|
|
initOs(void)
|
|
{ GET_LD
|
|
|
|
GD->statistics.start_time = WallTime();
|
|
|
|
DEBUG(1, Sdprintf("OS:initExpand() ...\n"));
|
|
initExpand();
|
|
DEBUG(1, Sdprintf("OS:initEnviron() ...\n"));
|
|
initEnviron();
|
|
|
|
#ifdef __WINDOWS__
|
|
setPrologFlagMask(PLFLAG_FILE_CASE_PRESERVING);
|
|
#else
|
|
setPrologFlagMask(PLFLAG_FILE_CASE);
|
|
setPrologFlagMask(PLFLAG_FILE_CASE_PRESERVING);
|
|
#endif
|
|
|
|
DEBUG(1, Sdprintf("OS:done\n"));
|
|
|
|
succeed;
|
|
}
|
|
|
|
|
|
void
|
|
cleanupOs(void)
|
|
{ cleanupExpand();
|
|
}
|
|
|
|
|
|
/********************************
|
|
* OS ERRORS *
|
|
*********************************/
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
char *OsError()
|
|
Return a char *, holding a description of the last OS call error.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
char *
|
|
OsError(void)
|
|
{
|
|
#ifdef HAVE_STRERROR
|
|
#ifdef __WINDOWS__
|
|
return strerror(_xos_errno());
|
|
#else
|
|
return strerror(errno);
|
|
#endif
|
|
#else /*HAVE_STRERROR*/
|
|
static char errmsg[64];
|
|
|
|
#ifdef __unix__
|
|
extern int sys_nerr;
|
|
#if !EMX
|
|
extern char *sys_errlist[];
|
|
#endif
|
|
extern int errno;
|
|
|
|
if ( errno < sys_nerr )
|
|
return sys_errlist[errno];
|
|
#endif
|
|
|
|
Ssprintf(errmsg, "Unknown Error (%d)", errno);
|
|
return errmsg;
|
|
#endif /*HAVE_STRERROR*/
|
|
}
|
|
|
|
/********************************
|
|
* PROCESS CHARACTERISTICS *
|
|
*********************************/
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
double CpuTime(cputime_kind)
|
|
|
|
Returns a floating point number, representing the amount of (user)
|
|
CPU-seconds used by the process Prolog is in. For systems that do
|
|
not allow you to obtain this information you may wish to return
|
|
elapsed time since Prolog was started, as this function is used to
|
|
by consult/1 and time/1 to determine the amount of CPU time used to
|
|
consult a file or to execute a query.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
#ifdef HAVE_CLOCK_GETTIME
|
|
#define timespec_to_double(ts) \
|
|
((double)(ts).tv_sec + (double)(ts).tv_nsec/(double)1000000000.0)
|
|
#endif
|
|
|
|
#ifndef __WINDOWS__ /* defined in pl-nt.c */
|
|
|
|
#ifdef HAVE_TIMES
|
|
#include <sys/times.h>
|
|
|
|
#if defined(_SC_CLK_TCK)
|
|
#define Hz ((int)sysconf(_SC_CLK_TCK))
|
|
#else
|
|
#ifdef HZ
|
|
# define Hz HZ
|
|
#else
|
|
# define Hz 60 /* if nothing better: guess */
|
|
#endif
|
|
#endif /*_SC_CLK_TCK*/
|
|
#endif /*HAVE_TIMES*/
|
|
|
|
|
|
double
|
|
CpuTime(cputime_kind which)
|
|
{
|
|
#if defined(HAVE_CLOCK_GETTIME) && defined(CLOCK_PROCESS_CPUTIME_ID)
|
|
#define CPU_TIME_DONE
|
|
struct timespec ts;
|
|
(void)which;
|
|
|
|
if ( clock_gettime(CLOCK_PROCESS_CPUTIME_ID, &ts) == 0 )
|
|
return timespec_to_double(ts);
|
|
return 0.0;
|
|
#endif
|
|
|
|
#if !defined(CPU_TIME_DONE) && defined(HAVE_TIMES)
|
|
#define CPU_TIME_DONE
|
|
struct tms t;
|
|
double used;
|
|
static int MTOK_got_hz = FALSE;
|
|
static double MTOK_hz;
|
|
|
|
if ( !MTOK_got_hz )
|
|
{ MTOK_hz = (double) Hz;
|
|
MTOK_got_hz++;
|
|
}
|
|
times(&t);
|
|
|
|
switch( which )
|
|
{ case CPU_USER:
|
|
used = (double) t.tms_utime / MTOK_hz;
|
|
break;
|
|
case CPU_SYSTEM:
|
|
default: /* make compiler happy */
|
|
used = (double) t.tms_stime / MTOK_hz;
|
|
}
|
|
|
|
if ( isnan(used) ) /* very dubious, but this */
|
|
used = 0.0; /* happens when running under GDB */
|
|
|
|
return used;
|
|
#endif
|
|
|
|
#if !defined(CPU_TIME_DONE)
|
|
(void)which;
|
|
|
|
return 0.0;
|
|
#endif
|
|
}
|
|
|
|
#endif /*__WINDOWS__*/
|
|
|
|
|
|
double
|
|
WallTime(void)
|
|
{ double stime;
|
|
|
|
#if HAVE_CLOCK_GETTIME
|
|
struct timespec tp;
|
|
|
|
clock_gettime(CLOCK_REALTIME, &tp);
|
|
stime = timespec_to_double(tp);
|
|
#else
|
|
#ifdef HAVE_GETTIMEOFDAY
|
|
struct timeval tp;
|
|
|
|
gettimeofday(&tp, NULL);
|
|
stime = (double)tp.tv_sec + (double)tp.tv_usec/1000000.0;
|
|
#else
|
|
#ifdef HAVE_FTIME
|
|
struct timeb tb;
|
|
|
|
ftime(&tb);
|
|
stime = (double)tb.time + (double)tb.millitm/1000.0;
|
|
#else
|
|
stime = (double)time((time_t *)NULL);
|
|
#endif
|
|
#endif
|
|
#endif
|
|
|
|
return stime;
|
|
}
|
|
|
|
/*******************************
|
|
* FEATURES *
|
|
*******************************/
|
|
|
|
#ifndef __WINDOWS__ /* Windows version in pl-nt.c */
|
|
|
|
#ifdef HAVE_SC_NPROCESSORS_CONF
|
|
static int
|
|
CpuCount()
|
|
{
|
|
return sysconf(_SC_NPROCESSORS_CONF);
|
|
}
|
|
#else
|
|
|
|
#ifdef PROCFS_CPUINFO
|
|
static int
|
|
CpuCount()
|
|
{ FILE *fd = fopen("/proc/cpuinfo", "r");
|
|
|
|
if ( fd )
|
|
{ char buf[256];
|
|
int count = 0;
|
|
|
|
while(fgets(buf, sizeof(buf)-1, fd))
|
|
{ char *vp;
|
|
|
|
if ( (vp = strchr(buf, ':')) )
|
|
{ char *en;
|
|
|
|
for(en=vp; en > buf && en[-1] <= ' '; en--)
|
|
;
|
|
*en = EOS;
|
|
DEBUG(2, Sdprintf("Got %s = %s\n", buf, vp+2));
|
|
if ( streq("processor", buf) && isDigit(vp[2]) )
|
|
{ int cpu = atoi(vp+2);
|
|
|
|
if ( cpu+1 > count )
|
|
count = cpu+1;
|
|
}
|
|
}
|
|
}
|
|
|
|
fclose(fd);
|
|
return count;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
#else /*PROCFS_CPUINFO*/
|
|
|
|
#ifdef HAVE_SYSCTLBYNAME /* MacOS X */
|
|
|
|
#include <sys/param.h>
|
|
#include <sys/sysctl.h>
|
|
|
|
int
|
|
CpuCount(void)
|
|
{ int count ;
|
|
size_t size=sizeof(count) ;
|
|
|
|
if ( sysctlbyname("hw.ncpu", &count, &size, NULL, 0) )
|
|
return 0;
|
|
|
|
return count;
|
|
}
|
|
|
|
#else
|
|
|
|
#define CpuCount() 0
|
|
|
|
#endif /*sysctlbyname*/
|
|
|
|
#endif /*PROCFS_CPUINFO*/
|
|
|
|
#endif /*HAVE_SC_NPROCESSORS_CONF*/
|
|
|
|
|
|
void
|
|
setOSPrologFlags(void)
|
|
{ int cpu_count = CpuCount();
|
|
|
|
if ( cpu_count > 0 )
|
|
PL_set_prolog_flag("cpu_count", PL_INTEGER, cpu_count);
|
|
}
|
|
#endif
|
|
|
|
/*******************************
|
|
* MEMORY *
|
|
*******************************/
|
|
|
|
uintptr_t
|
|
UsedMemory(void)
|
|
{ //GET_LD
|
|
|
|
#if defined(HAVE_GETRUSAGE) && defined(HAVE_RU_IDRSS)
|
|
struct rusage usage;
|
|
|
|
if ( getrusage(RUSAGE_SELF, &usage) == 0 &&
|
|
usage.ru_idrss )
|
|
{ return usage.ru_idrss; /* total unshared data */
|
|
}
|
|
#endif
|
|
|
|
return (usedStack(global) +
|
|
usedStack(local) +
|
|
usedStack(trail));
|
|
}
|
|
|
|
|
|
uintptr_t
|
|
FreeMemory(void)
|
|
{
|
|
#if defined(HAVE_GETRLIMIT) && defined(RLIMIT_DATA)
|
|
uintptr_t used = UsedMemory();
|
|
struct rlimit limit;
|
|
|
|
if ( getrlimit(RLIMIT_DATA, &limit) == 0 )
|
|
return limit.rlim_cur - used;
|
|
#endif
|
|
|
|
return 0L;
|
|
}
|
|
|
|
|
|
/********************************
|
|
* ARITHMETIC *
|
|
*********************************/
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
uint64_t _PL_Random()
|
|
|
|
Return a random number. Used for arithmetic only. More trouble. On
|
|
some systems (__WINDOWS__) the seed of rand() is thread-local, while on
|
|
others it is global. We appear to have the choice between
|
|
|
|
# srand()/rand()
|
|
Differ in MT handling, often bad distribution
|
|
|
|
# srandom()/random()
|
|
Not portable, not MT-Safe but much better distribution
|
|
|
|
# drand48() and friends
|
|
Depreciated according to Linux manpage, suggested by Solaris
|
|
manpage.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
void
|
|
setRandom(unsigned int *seedp)
|
|
{ unsigned int seed;
|
|
|
|
if ( seedp )
|
|
{ seed = *seedp;
|
|
} else
|
|
{
|
|
#ifdef __WINDOWS__
|
|
seed = (unsigned int)GetTickCount();
|
|
#else
|
|
#ifdef HAVE_GETTIMEOFDAY
|
|
struct timeval tp;
|
|
|
|
gettimeofday(&tp, NULL);
|
|
seed = (unsigned int)(tp.tv_sec + tp.tv_usec);
|
|
#else
|
|
seed = (unsigned int)time((time_t *) NULL);
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
#if HAVE_SRANDOM
|
|
srandom(seed);
|
|
#elif HAVE_SRAND
|
|
srand(seed);
|
|
#endif
|
|
}
|
|
|
|
uint64_t
|
|
_PL_Random(void)
|
|
{ GET_LD
|
|
|
|
if ( !LD->os.rand_initialised )
|
|
{ setRandom(NULL);
|
|
LD->os.rand_initialised = TRUE;
|
|
}
|
|
|
|
#ifdef HAVE_RANDOM
|
|
{ uint64_t l = random();
|
|
|
|
l ^= (uint64_t)random()<<15;
|
|
l ^= (uint64_t)random()<<30;
|
|
l ^= (uint64_t)random()<<45;
|
|
|
|
return l;
|
|
}
|
|
#else
|
|
{ uint64_t l = rand(); /* 0<n<2^15-1 */
|
|
|
|
l ^= (uint64_t)rand()<<15;
|
|
l ^= (uint64_t)rand()<<30;
|
|
l ^= (uint64_t)rand()<<45;
|
|
|
|
return l;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
/********************************
|
|
* FILES *
|
|
*********************************/
|
|
|
|
/* (Everything you always wanted to know about files ...) */
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
Generation and administration of temporary files. Currently only used
|
|
by the foreign language linker. It might be useful to make a predicate
|
|
available to the Prolog user based on these functions. These functions
|
|
are in this module as non-UNIX OS probably don't have getpid() or put
|
|
temporaries on /tmp.
|
|
|
|
atom_t TemporaryFile(const char *id, int *fdp)
|
|
|
|
The return value of this call is an atom, whose string represents
|
|
the path name of a unique file that can be used as temporary file.
|
|
`id' is a char * that can be used to make it easier to identify the
|
|
file as a specific kind of SWI-Prolog intermediate file.
|
|
|
|
void RemoveTemporaryFiles()
|
|
|
|
Remove all temporary files. This function should be aware of the
|
|
fact that some of the file names generated by TemporaryFile() might
|
|
not be created at all, or might already have been deleted.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
#ifndef DEFTMPDIR
|
|
#ifdef __WINDOWS__
|
|
#define DEFTMPDIR "c:/tmp"
|
|
#else
|
|
#define DEFTMPDIR "/tmp"
|
|
#endif
|
|
#endif
|
|
|
|
static int
|
|
free_tmp_symbol(Symbol s)
|
|
{ int rc;
|
|
atom_t tname = (atom_t)s->name;
|
|
PL_chars_t txt;
|
|
|
|
get_atom_text(tname, &txt);
|
|
PL_mb_text(&txt, REP_FN);
|
|
rc = RemoveFile(txt.text.t);
|
|
PL_free_text(&txt);
|
|
|
|
PL_unregister_atom(tname);
|
|
return rc;
|
|
}
|
|
|
|
|
|
static void
|
|
void_free_tmp_symbol(Symbol s)
|
|
{ (void)free_tmp_symbol(s);
|
|
}
|
|
|
|
|
|
#ifndef O_EXCL
|
|
#define O_EXCL 0
|
|
#endif
|
|
#ifndef O_BINARY
|
|
#define O_BINARY 0
|
|
#endif
|
|
|
|
atom_t
|
|
TemporaryFile(const char *id, int *fdp)
|
|
{ char temp[MAXPATHLEN];
|
|
static char *tmpdir = NULL;
|
|
atom_t tname;
|
|
int retries = 0;
|
|
|
|
if ( !tmpdir )
|
|
{ LOCK();
|
|
if ( !tmpdir )
|
|
{ char envbuf[MAXPATHLEN];
|
|
char *td;
|
|
|
|
if ( (td = Getenv("TEMP", envbuf, sizeof(envbuf))) ||
|
|
(td = Getenv("TMP", envbuf, sizeof(envbuf))) )
|
|
tmpdir = strdup(td);
|
|
else
|
|
tmpdir = DEFTMPDIR;
|
|
}
|
|
UNLOCK();
|
|
}
|
|
|
|
retry:
|
|
#ifdef __unix__
|
|
{ static int MTOK_temp_counter = 0;
|
|
const char *sep = id[0] ? "_" : "";
|
|
|
|
Ssprintf(temp, "%s/pl_%s%s%d_%d",
|
|
tmpdir, id, sep, (int) getpid(), MTOK_temp_counter++);
|
|
}
|
|
#endif
|
|
|
|
#ifdef __WINDOWS__
|
|
{ char *tmp;
|
|
static int temp_counter = 0;
|
|
|
|
#ifdef __LCC__
|
|
if ( (tmp = tmpnam(NULL)) )
|
|
#else
|
|
if ( (tmp = _tempnam(tmpdir, id)) )
|
|
#endif
|
|
{ PrologPath(tmp, temp, sizeof(temp));
|
|
} else
|
|
{ const char *sep = id[0] ? "_" : "";
|
|
|
|
Ssprintf(temp, "%s/pl_%s%s%d", tmpdir, id, sep, temp_counter++);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
if ( fdp )
|
|
{ int fd;
|
|
|
|
if ( (fd=open(temp, O_CREAT|O_EXCL|O_WRONLY|O_BINARY, 0600)) < 0 )
|
|
{ if ( ++retries < 10000 )
|
|
goto retry;
|
|
else
|
|
return NULL_ATOM;
|
|
}
|
|
|
|
*fdp = fd;
|
|
}
|
|
|
|
tname = PL_new_atom(temp); /* locked: ok! */
|
|
|
|
LOCK();
|
|
if ( !GD->os.tmp_files )
|
|
{ GD->os.tmp_files = newHTable(4);
|
|
GD->os.tmp_files->free_symbol = void_free_tmp_symbol;
|
|
}
|
|
UNLOCK();
|
|
|
|
addHTable(GD->os.tmp_files, (void*)tname, (void*)TRUE);
|
|
|
|
return tname;
|
|
}
|
|
|
|
|
|
int
|
|
DeleteTemporaryFile(atom_t name)
|
|
{ int rc = FALSE;
|
|
|
|
if ( GD->os.tmp_files )
|
|
{ LOCK();
|
|
if ( GD->os.tmp_files && GD->os.tmp_files->size > 0 )
|
|
{ Symbol s = lookupHTable(GD->os.tmp_files, (void*)name);
|
|
|
|
if ( s )
|
|
{ rc = free_tmp_symbol(s);
|
|
deleteSymbolHTable(GD->os.tmp_files, s);
|
|
}
|
|
}
|
|
UNLOCK();
|
|
}
|
|
|
|
return rc;
|
|
}
|
|
|
|
|
|
void
|
|
RemoveTemporaryFiles(void)
|
|
{ LOCK();
|
|
if ( GD->os.tmp_files )
|
|
{ Table t = GD->os.tmp_files;
|
|
|
|
GD->os.tmp_files = NULL;
|
|
UNLOCK();
|
|
destroyHTable(t);
|
|
} else
|
|
{ UNLOCK();
|
|
}
|
|
}
|
|
|
|
|
|
#if O_HPFS
|
|
|
|
/* Conversion rules Prolog <-> OS/2 (using HPFS)
|
|
/ <-> \
|
|
/x:/ <-> x:\ (embedded drive letter)
|
|
No length restrictions up to MAXPATHLEN, no case conversions.
|
|
*/
|
|
|
|
char *
|
|
PrologPath(char *ospath, char *path, size_t len)
|
|
{ char *s = ospath, *p = path;
|
|
int limit = len-1;
|
|
|
|
if (isLetter(s[0]) && s[1] == ':')
|
|
{ *p++ = '/';
|
|
*p++ = *s++;
|
|
*p++ = *s++;
|
|
limit -= 3;
|
|
}
|
|
for(; *s && limit; s++, p++, limit--)
|
|
*p = (*s == '\\' ? '/' : makeLower(*s));
|
|
*p = EOS;
|
|
|
|
return path;
|
|
}
|
|
|
|
|
|
char *
|
|
OsPath(const char *plpath, char *path)
|
|
{ const char *s = plpath, *p = path;
|
|
int limit = MAXPATHLEN-1;
|
|
|
|
if ( s[0] == '/' && isLetter(s[1]) && s[2] == ':') /* embedded drive letter*/
|
|
{ s++;
|
|
*p++ = *s++;
|
|
*p++ = *s++;
|
|
if ( *s != '/' )
|
|
*p++ = '\\';
|
|
limit -= 2;
|
|
}
|
|
|
|
for(; *s && limit; s++, p++, limit--)
|
|
*p = (*s == '/' ? '\\' : *s);
|
|
if ( p[-1] == '\\' && p > path )
|
|
p--;
|
|
*p = EOS;
|
|
|
|
return path;
|
|
}
|
|
#endif /* O_HPFS */
|
|
|
|
#ifdef __unix__
|
|
char *
|
|
PrologPath(const char *p, char *buf, size_t len)
|
|
{ strncpy(buf, p, len);
|
|
|
|
return buf;
|
|
}
|
|
|
|
char *
|
|
OsPath(const char *p, char *buf)
|
|
{ strcpy(buf, p);
|
|
|
|
return buf;
|
|
}
|
|
#endif /*__unix__*/
|
|
|
|
#if O_XOS
|
|
char *
|
|
PrologPath(const char *p, char *buf, size_t len)
|
|
{ GET_LD
|
|
int flags = (truePrologFlag(PLFLAG_FILE_CASE) ? 0 : XOS_DOWNCASE);
|
|
|
|
return _xos_canonical_filename(p, buf, len, flags);
|
|
}
|
|
|
|
char *
|
|
OsPath(const char *p, char *buf)
|
|
{ strcpy(buf, p);
|
|
|
|
return buf;
|
|
}
|
|
#endif /* O_XOS */
|
|
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
char *AbsoluteFile(const char *file, char *path)
|
|
|
|
Expand a file specification to a system-wide unique description of
|
|
the file that can be passed to the file functions that take a path
|
|
as argument. Path should refer to the same file, regardless of the
|
|
current working directory. On Unix absolute file names are used
|
|
for this purpose.
|
|
|
|
This function is based on a similar (primitive) function in
|
|
Edinburgh C-Prolog.
|
|
|
|
char *BaseName(path)
|
|
char *path;
|
|
|
|
Return the basic file name for a file having path `path'.
|
|
|
|
char *DirName(const char *path, char *dir)
|
|
|
|
Return the directory name for a file having path `path'.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
#if defined(HAVE_SYMLINKS) && (defined(HAVE_STAT) || defined(__unix__))
|
|
#define O_CANONISE_DIRS
|
|
|
|
struct canonical_dir
|
|
{ char * name; /* name of directory */
|
|
char * canonical; /* canonical name of directory */
|
|
dev_t device; /* device number */
|
|
ino_t inode; /* inode number */
|
|
CanonicalDir next; /* next in chain */
|
|
};
|
|
|
|
#define canonical_dirlist (GD->os._canonical_dirlist)
|
|
|
|
forwards char *canoniseDir(char *);
|
|
#endif /*O_CANONISE_DIRS*/
|
|
|
|
static void
|
|
initExpand(void)
|
|
{
|
|
#ifdef O_CANONISE_DIRS
|
|
char *dir;
|
|
char *cpaths;
|
|
#endif
|
|
|
|
GD->paths.CWDdir = NULL;
|
|
GD->paths.CWDlen = 0;
|
|
|
|
#ifdef O_CANONISE_DIRS
|
|
{ char envbuf[MAXPATHLEN];
|
|
|
|
if ( (cpaths = Getenv("CANONICAL_PATHS", envbuf, sizeof(envbuf))) )
|
|
{ char buf[MAXPATHLEN];
|
|
|
|
while(*cpaths)
|
|
{ char *e;
|
|
|
|
if ( (e = strchr(cpaths, ':')) )
|
|
{ int l = e-cpaths;
|
|
|
|
strncpy(buf, cpaths, l);
|
|
buf[l] = EOS;
|
|
cpaths += l+1;
|
|
canoniseDir(buf);
|
|
} else
|
|
{ canoniseDir(cpaths);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
if ( (dir = Getenv("HOME", envbuf, sizeof(envbuf))) ) canoniseDir(dir);
|
|
if ( (dir = Getenv("PWD", envbuf, sizeof(envbuf))) ) canoniseDir(dir);
|
|
if ( (dir = Getenv("CWD", envbuf, sizeof(envbuf))) ) canoniseDir(dir);
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#ifdef O_CANONISE_DIRS
|
|
|
|
static void
|
|
cleanupExpand(void)
|
|
{ CanonicalDir dn = canonical_dirlist, next;
|
|
|
|
canonical_dirlist = NULL;
|
|
for( ; dn; dn = next )
|
|
{ next = dn->next;
|
|
if ( dn->canonical && dn->canonical != dn->name )
|
|
remove_string(dn->canonical);
|
|
remove_string(dn->name);
|
|
PL_free(dn);
|
|
}
|
|
if ( GD->paths.CWDdir )
|
|
{ remove_string(GD->paths.CWDdir);
|
|
GD->paths.CWDdir = NULL;
|
|
GD->paths.CWDlen = 0;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
registerParentDirs(const char *path)
|
|
{ const char *e = path + strlen(path);
|
|
|
|
while(e>path)
|
|
{ char dirname[MAXPATHLEN];
|
|
char tmp[MAXPATHLEN];
|
|
CanonicalDir d;
|
|
statstruct buf;
|
|
|
|
for(e--; *e != '/' && e > path + 1; e-- )
|
|
;
|
|
|
|
strncpy(dirname, path, e-path);
|
|
dirname[e-path] = EOS;
|
|
|
|
for(d = canonical_dirlist; d; d = d->next)
|
|
{ if ( streq(d->name, dirname) )
|
|
return;
|
|
}
|
|
|
|
if ( statfunc(OsPath(dirname, tmp), &buf) == 0 )
|
|
{ CanonicalDir dn = PL_malloc(sizeof(*dn));
|
|
|
|
dn->name = store_string(dirname);
|
|
dn->inode = buf.st_ino;
|
|
dn->device = buf.st_dev;
|
|
dn->canonical = dn->name;
|
|
dn->next = canonical_dirlist;
|
|
canonical_dirlist = dn;
|
|
|
|
DEBUG(1, Sdprintf("Registered canonical dir %s\n", dirname));
|
|
} else
|
|
return;
|
|
}
|
|
}
|
|
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
verify_entry() verifies the path cache for this path is still safe. If
|
|
not it updates the cache and returns FALSE.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
static int
|
|
verify_entry(CanonicalDir d)
|
|
{ char tmp[MAXPATHLEN];
|
|
statstruct buf;
|
|
|
|
if ( statfunc(OsPath(d->canonical, tmp), &buf) == 0 )
|
|
{ if ( d->inode == buf.st_ino &&
|
|
d->device == buf.st_dev )
|
|
return TRUE;
|
|
|
|
DEBUG(1, Sdprintf("%s: inode/device changed\n", d->canonical));
|
|
|
|
d->inode = buf.st_ino;
|
|
d->device = buf.st_dev;
|
|
return TRUE;
|
|
} else
|
|
{ DEBUG(1, Sdprintf("%s: no longer exists\n", d->canonical));
|
|
|
|
if ( d == canonical_dirlist )
|
|
{ canonical_dirlist = d->next;
|
|
} else
|
|
{ CanonicalDir cd;
|
|
|
|
for(cd=canonical_dirlist; cd; cd=cd->next)
|
|
{ if ( cd->next == d )
|
|
{ cd->next = d->next;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
remove_string(d->name);
|
|
if ( d->canonical != d->name )
|
|
remove_string(d->canonical);
|
|
PL_free(d);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static char *
|
|
canoniseDir(char *path)
|
|
{ CanonicalDir d, next;
|
|
statstruct buf;
|
|
char tmp[MAXPATHLEN];
|
|
|
|
DEBUG(1, Sdprintf("canoniseDir(%s) --> ", path));
|
|
|
|
for(d = canonical_dirlist; d; d = next)
|
|
{ next = d->next;
|
|
|
|
if ( streq(d->name, path) && verify_entry(d) )
|
|
{ if ( d->name != d->canonical )
|
|
strcpy(path, d->canonical);
|
|
|
|
DEBUG(1, Sdprintf("(lookup) %s\n", path));
|
|
return path;
|
|
}
|
|
}
|
|
|
|
/* we need to use malloc() here */
|
|
/* because allocHeapOrHalt() only ensures */
|
|
/* alignment for `word', and inode_t */
|
|
/* is sometimes bigger! */
|
|
|
|
if ( statfunc(OsPath(path, tmp), &buf) == 0 )
|
|
{ CanonicalDir dn = PL_malloc(sizeof(*dn));
|
|
char dirname[MAXPATHLEN];
|
|
char *e = path + strlen(path);
|
|
|
|
dn->name = store_string(path);
|
|
dn->inode = buf.st_ino;
|
|
dn->device = buf.st_dev;
|
|
|
|
do
|
|
{ strncpy(dirname, path, e-path);
|
|
dirname[e-path] = EOS;
|
|
if ( statfunc(OsPath(dirname, tmp), &buf) < 0 )
|
|
break;
|
|
|
|
DEBUG(2, Sdprintf("Checking %s (dev=%d,ino=%d)\n",
|
|
dirname, buf.st_dev, buf.st_ino));
|
|
|
|
for(d = canonical_dirlist; d; d = next)
|
|
{ next = d->next;
|
|
|
|
if ( d->inode == buf.st_ino && d->device == buf.st_dev &&
|
|
verify_entry(d) )
|
|
{ DEBUG(2, Sdprintf("Hit with %s (dev=%d,ino=%d)\n",
|
|
d->canonical, d->device, d->inode));
|
|
|
|
strcpy(dirname, d->canonical);
|
|
strcat(dirname, e);
|
|
strcpy(path, dirname);
|
|
dn->canonical = store_string(path);
|
|
dn->next = canonical_dirlist;
|
|
canonical_dirlist = dn;
|
|
DEBUG(1, Sdprintf("(replace) %s\n", path));
|
|
registerParentDirs(path);
|
|
return path;
|
|
}
|
|
}
|
|
|
|
for(e--; *e != '/' && e > path + 1; e-- )
|
|
;
|
|
} while( e > path );
|
|
|
|
dn->canonical = dn->name;
|
|
dn->next = canonical_dirlist;
|
|
canonical_dirlist = dn;
|
|
|
|
DEBUG(1, Sdprintf("(new, existing) %s\n", path));
|
|
registerParentDirs(path);
|
|
return path;
|
|
}
|
|
|
|
DEBUG(1, Sdprintf("(nonexisting) %s\n", path));
|
|
return path;
|
|
}
|
|
|
|
#else
|
|
|
|
#define canoniseDir(d)
|
|
|
|
static void
|
|
cleanupExpand(void)
|
|
{
|
|
}
|
|
|
|
#endif /*O_CANONISE_DIRS*/
|
|
|
|
|
|
char *
|
|
canoniseFileName(char *path)
|
|
{ char *out = path, *in = path, *start = path;
|
|
tmp_buffer saveb;
|
|
|
|
#ifdef O_HASDRIVES /* C: */
|
|
if ( in[1] == ':' && isLetter(in[0]) )
|
|
{ in += 2;
|
|
|
|
out = start = in;
|
|
}
|
|
#ifdef __MINGW32__ /* /c/ in MINGW is the same as c: */
|
|
else if ( in[0] == '/' && isLetter(in[1]) &&
|
|
in[2] == '/' )
|
|
{
|
|
out[0] = in[1];
|
|
out[1] = ':';
|
|
in += 3;
|
|
out = start = in;
|
|
}
|
|
#endif
|
|
#endif
|
|
|
|
#ifdef O_HASSHARES /* //host/ */
|
|
if ( in[0] == '/' && in[1] == '/' && isAlpha(in[2]) )
|
|
{ char *s;
|
|
|
|
for(s = in+3; *s && (isAlpha(*s) || *s == '-' || *s == '.'); s++)
|
|
;
|
|
if ( *s == '/' )
|
|
{ in = out = s+1;
|
|
start = in-1;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
while( in[0] == '/' && in[1] == '.' && in[2] == '.' && in[3] == '/' )
|
|
in += 3;
|
|
while( in[0] == '.' && in[1] == '/' )
|
|
in += 2;
|
|
if ( in[0] == '/' )
|
|
*out++ = '/';
|
|
initBuffer(&saveb);
|
|
addBuffer(&saveb, out, char*);
|
|
|
|
while(*in)
|
|
{ if (*in == '/')
|
|
{
|
|
again:
|
|
if ( *in )
|
|
{ while( in[1] == '/' ) /* delete multiple / */
|
|
in++;
|
|
if ( in[1] == '.' )
|
|
{ if ( in[2] == '/' ) /* delete /./ */
|
|
{ in += 2;
|
|
goto again;
|
|
}
|
|
if ( in[2] == EOS ) /* delete trailing /. */
|
|
{ *out = EOS;
|
|
goto out;
|
|
}
|
|
if ( in[2] == '.' && (in[3] == '/' || in[3] == EOS) )
|
|
{ if ( !isEmptyBuffer(&saveb) ) /* delete /foo/../ */
|
|
{ out = popBuffer(&saveb, char*);
|
|
in += 3;
|
|
if ( in[0] == EOS && out > start+1 )
|
|
{ out[-1] = EOS; /* delete trailing / */
|
|
goto out;
|
|
}
|
|
goto again;
|
|
} else if ( start[0] == '/' && out == start+1 )
|
|
{ in += 3;
|
|
goto again;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
if ( *in )
|
|
in++;
|
|
if ( out > path && out[-1] != '/' )
|
|
*out++ = '/';
|
|
addBuffer(&saveb, out, char*);
|
|
} else
|
|
*out++ = *in++;
|
|
}
|
|
*out++ = *in++;
|
|
|
|
out:
|
|
discardBuffer(&saveb);
|
|
|
|
return path;
|
|
}
|
|
|
|
|
|
static char *
|
|
utf8_strlwr(char *s)
|
|
{ char tmp[MAXPATHLEN];
|
|
char *o, *i;
|
|
|
|
strcpy(tmp, s);
|
|
for(i=tmp, o=s; *i; )
|
|
{ int c;
|
|
|
|
i = utf8_get_char(i, &c);
|
|
c = towlower((wint_t)c);
|
|
o = utf8_put_char(o, c);
|
|
}
|
|
*o = EOS;
|
|
|
|
return s;
|
|
}
|
|
|
|
|
|
char *
|
|
canonisePath(char *path)
|
|
{ GET_LD
|
|
|
|
if ( !truePrologFlag(PLFLAG_FILE_CASE) )
|
|
utf8_strlwr(path);
|
|
|
|
canoniseFileName(path);
|
|
|
|
#ifdef O_CANONISE_DIRS
|
|
{ char *e;
|
|
char dirname[MAXPATHLEN];
|
|
size_t plen = strlen(path);
|
|
|
|
if ( plen > 0 )
|
|
{ e = path + plen - 1;
|
|
for( ; *e != '/' && e > path; e-- )
|
|
;
|
|
strncpy(dirname, path, e-path);
|
|
dirname[e-path] = EOS;
|
|
canoniseDir(dirname);
|
|
strcat(dirname, e);
|
|
strcpy(path, dirname);
|
|
}
|
|
}
|
|
#endif
|
|
|
|
return path;
|
|
}
|
|
|
|
|
|
static char *
|
|
takeWord(const char **string, char *wrd, int maxlen)
|
|
{ const char *s = *string;
|
|
char *q = wrd;
|
|
int left = maxlen-1;
|
|
|
|
while( isAlpha(*s) || *s == '_' )
|
|
{ if ( --left < 0 )
|
|
{ PL_error(NULL, 0, NULL, ERR_REPRESENTATION,
|
|
ATOM_max_variable_length);
|
|
return NULL;
|
|
}
|
|
*q++ = *s++;
|
|
}
|
|
*q = EOS;
|
|
|
|
*string = s;
|
|
return wrd;
|
|
}
|
|
|
|
|
|
char *
|
|
expandVars(const char *pattern, char *expanded, int maxlen)
|
|
{ GET_LD
|
|
int size = 0;
|
|
char wordbuf[MAXPATHLEN];
|
|
char *rc = expanded;
|
|
|
|
if ( *pattern == '~' )
|
|
{ char *user;
|
|
char *value;
|
|
int l;
|
|
|
|
pattern++;
|
|
user = takeWord(&pattern, wordbuf, sizeof(wordbuf));
|
|
|
|
LOCK();
|
|
|
|
if ( user[0] == EOS ) /* ~/bla */
|
|
{
|
|
#ifdef O_XOS
|
|
value = _xos_home();
|
|
#else /*O_XOS*/
|
|
if ( !(value = GD->os.myhome) )
|
|
{ char envbuf[MAXPATHLEN];
|
|
|
|
if ( (value = Getenv("HOME", envbuf, sizeof(envbuf))) &&
|
|
(value = PrologPath(value, wordbuf, sizeof(wordbuf))) )
|
|
{ GD->os.myhome = store_string(value);
|
|
} else
|
|
{ value = GD->os.myhome = store_string("/");
|
|
}
|
|
}
|
|
#endif /*O_XOS*/
|
|
} else /* ~fred */
|
|
#ifdef HAVE_GETPWNAM
|
|
{ struct passwd *pwent;
|
|
|
|
if ( GD->os.fred && streq(GD->os.fred, user) )
|
|
{ value = GD->os.fredshome;
|
|
} else
|
|
{ if ( !(pwent = getpwnam(user)) )
|
|
{ if ( truePrologFlag(PLFLAG_FILEERRORS) )
|
|
{ term_t name = PL_new_term_ref();
|
|
|
|
PL_put_atom_chars(name, user);
|
|
PL_error(NULL, 0, NULL, ERR_EXISTENCE, ATOM_user, name);
|
|
}
|
|
UNLOCK();
|
|
fail;
|
|
}
|
|
if ( GD->os.fred )
|
|
remove_string(GD->os.fred);
|
|
if ( GD->os.fredshome )
|
|
remove_string(GD->os.fredshome);
|
|
|
|
GD->os.fred = store_string(user);
|
|
value = GD->os.fredshome = store_string(pwent->pw_dir);
|
|
}
|
|
}
|
|
#else
|
|
{ if ( truePrologFlag(PLFLAG_FILEERRORS) )
|
|
PL_error(NULL, 0, NULL, ERR_NOT_IMPLEMENTED, "user_info");
|
|
|
|
UNLOCK();
|
|
fail;
|
|
}
|
|
#endif
|
|
size += (l = (int) strlen(value));
|
|
if ( size+1 >= maxlen )
|
|
{ PL_error(NULL, 0, NULL, ERR_REPRESENTATION, ATOM_max_path_length);
|
|
return NULL;
|
|
}
|
|
strcpy(expanded, value);
|
|
expanded += l;
|
|
UNLOCK();
|
|
|
|
/* ~/ should not become // */
|
|
if ( expanded[-1] == '/' && pattern[0] == '/' )
|
|
pattern++;
|
|
}
|
|
|
|
for( ;; )
|
|
{ int c = *pattern++;
|
|
|
|
switch( c )
|
|
{ case EOS:
|
|
break;
|
|
case '$':
|
|
{ char envbuf[MAXPATHLEN];
|
|
char *var;
|
|
char *value, ch;
|
|
int l, i;
|
|
|
|
if (pattern[0] == '{') {
|
|
pattern++;
|
|
for (i = 0; i < sizeof(envbuf)-1; i++) {
|
|
if ((ch = *pattern++) == '}')
|
|
break;
|
|
envbuf[i] = ch;
|
|
}
|
|
envbuf[i] = '\0';
|
|
var = envbuf;
|
|
} else {
|
|
var = takeWord(&pattern, wordbuf, sizeof(wordbuf));
|
|
}
|
|
|
|
if ( var[0] == EOS )
|
|
goto def;
|
|
LOCK();
|
|
value = Getenv(var, envbuf, sizeof(envbuf));
|
|
if ( value == (char *) NULL )
|
|
{ if ( truePrologFlag(PLFLAG_FILEERRORS) )
|
|
{ term_t name = PL_new_term_ref();
|
|
|
|
PL_put_atom_chars(name, var);
|
|
PL_error(NULL, 0, NULL, ERR_EXISTENCE, ATOM_variable, name);
|
|
}
|
|
|
|
UNLOCK();
|
|
fail;
|
|
}
|
|
size += (l = (int)strlen(value));
|
|
if ( size+1 >= maxlen )
|
|
{ UNLOCK();
|
|
PL_error(NULL, 0, NULL, ERR_REPRESENTATION,
|
|
ATOM_max_path_length);
|
|
return NULL;
|
|
}
|
|
strcpy(expanded, value);
|
|
UNLOCK();
|
|
|
|
expanded += l;
|
|
|
|
continue;
|
|
}
|
|
default:
|
|
def:
|
|
size++;
|
|
if ( size+1 >= maxlen )
|
|
{ PL_error(NULL, 0, NULL, ERR_REPRESENTATION,
|
|
ATOM_max_path_length);
|
|
return NULL;
|
|
}
|
|
*expanded++ = c;
|
|
|
|
continue;
|
|
}
|
|
break;
|
|
}
|
|
|
|
if ( ++size >= maxlen )
|
|
{ PL_error(NULL, 0, NULL, ERR_REPRESENTATION,
|
|
ATOM_max_path_length);
|
|
return NULL;
|
|
}
|
|
|
|
*expanded = EOS;
|
|
|
|
return rc;
|
|
}
|
|
|
|
|
|
#ifdef O_HASDRIVES
|
|
|
|
#define IS_DIR_SEPARATOR(c) ((c) == '/' || (c) == '\\')
|
|
|
|
int
|
|
IsAbsolutePath(const char *p) /* /d:/ */
|
|
{ if ( p[0] == '/' && p[2] == ':' && isLetter(p[1]) &&
|
|
(p[3] == '/' || p[3] == '\0') )
|
|
succeed;
|
|
|
|
#ifdef __MINGW32__ /* /c/ in MINGW is the same as c: */
|
|
if ( p[0] == '/' && isLetter(p[1]) &&
|
|
(p[2] == '/' || p[2] == '\0') )
|
|
succeed;
|
|
#endif
|
|
|
|
if ( p[1] == ':' && isLetter(p[0]) && /* d:/ or d:\ */
|
|
(IS_DIR_SEPARATOR(p[2]) || p[2] == '\0') )
|
|
succeed;
|
|
|
|
#ifdef O_HASSHARES
|
|
if ( (p[0] == '/' && p[1] == '/') || /* //host/share */
|
|
(p[0] == '\\' && p[1] == '\\') ) /* \\host\share */
|
|
succeed;
|
|
#endif
|
|
|
|
fail;
|
|
}
|
|
|
|
|
|
static inline int
|
|
isDriveRelativePath(const char *p) /* '/...' */
|
|
{ return IS_DIR_SEPARATOR(p[0]) && !IsAbsolutePath(p);
|
|
}
|
|
|
|
#ifdef __WINDOWS__
|
|
#undef mkdir
|
|
#include <direct.h>
|
|
#define mkdir _xos_mkdir
|
|
#endif
|
|
|
|
static int
|
|
GetCurrentDriveLetter( void )
|
|
{
|
|
#ifdef OS2
|
|
return _getdrive();
|
|
#endif
|
|
#ifdef __WINDOWS__
|
|
return _getdrive() + 'a' - 1;
|
|
#endif
|
|
#ifdef __WATCOMC__
|
|
{ unsigned drive;
|
|
_dos_getdrive(&drive);
|
|
return = 'a' + drive - 1;
|
|
}
|
|
#endif
|
|
}
|
|
|
|
#else /*O_HASDRIVES*/
|
|
|
|
int
|
|
IsAbsolutePath(const char *p)
|
|
{ return p[0] == '/';
|
|
}
|
|
|
|
#endif /*O_HASDRIVES*/
|
|
|
|
#define isRelativePath(p) ( p[0] == '.' )
|
|
|
|
|
|
char *
|
|
AbsoluteFile(const char *spec, char *path)
|
|
{ GET_LD
|
|
char tmp[MAXPATHLEN];
|
|
char buf[MAXPATHLEN];
|
|
char *file = PrologPath(spec, buf, sizeof(buf));
|
|
|
|
if ( !file )
|
|
return (char *) NULL;
|
|
if ( truePrologFlag(PLFLAG_FILEVARS) )
|
|
{ if ( !(file = expandVars(buf, tmp, sizeof(tmp))) )
|
|
return (char *) NULL;
|
|
}
|
|
|
|
if ( IsAbsolutePath(file) )
|
|
{ strcpy(path, file);
|
|
|
|
return canonisePath(path);
|
|
}
|
|
|
|
#ifdef O_HASDRIVES
|
|
if ( isDriveRelativePath(file) ) /* /something --> d:/something */
|
|
{ if ((strlen(file) + 3) > MAXPATHLEN)
|
|
{ PL_error(NULL, 0, NULL, ERR_REPRESENTATION, ATOM_max_path_length);
|
|
return (char *) NULL;
|
|
}
|
|
path[0] = GetCurrentDriveLetter();
|
|
path[1] = ':';
|
|
strcpy(&path[2], file);
|
|
return canonisePath(path);
|
|
}
|
|
#endif /*O_HASDRIVES*/
|
|
|
|
if ( !PL_cwd(path, MAXPATHLEN) )
|
|
return NULL;
|
|
|
|
if ( (GD->paths.CWDlen + strlen(file) + 1) >= MAXPATHLEN )
|
|
{ PL_error(NULL, 0, NULL, ERR_REPRESENTATION, ATOM_max_path_length);
|
|
return (char *) NULL;
|
|
}
|
|
|
|
strcpy(path, GD->paths.CWDdir);
|
|
strcpy(&path[GD->paths.CWDlen], file);
|
|
|
|
return canonisePath(path);
|
|
}
|
|
|
|
|
|
void
|
|
PL_changed_cwd(void)
|
|
{ LOCK();
|
|
if ( GD->paths.CWDdir )
|
|
remove_string(GD->paths.CWDdir);
|
|
GD->paths.CWDdir = NULL;
|
|
GD->paths.CWDlen = 0;
|
|
UNLOCK();
|
|
}
|
|
|
|
static char *
|
|
cwd_unlocked(char *cwd, size_t cwdlen)
|
|
{ GET_LD
|
|
|
|
if ( GD->paths.CWDlen == 0 )
|
|
{ char buf[MAXPATHLEN];
|
|
char *rval;
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
On SunOs, getcwd() is using popen() to read the output of /bin/pwd. This
|
|
is slow and appears not to cooperate with profile/3. getwd() is supposed
|
|
to be implemented directly. What about other Unixes?
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
#if defined(HAVE_GETWD) && (defined(__sun__) || defined(__sun))
|
|
#undef HAVE_GETCWD
|
|
#endif
|
|
|
|
#if __ANDROID__
|
|
if (LOCAL_InAssetDir ) {
|
|
rval = strncpy(buf, LOCAL_InAssetDir, sizeof(buf));
|
|
} else
|
|
#endif
|
|
#if defined(HAVE_GETWD) && !defined(HAVE_GETCWD)
|
|
rval = getwd(buf);
|
|
#else
|
|
rval = getcwd(buf, sizeof(buf));
|
|
#endif
|
|
if ( !rval )
|
|
{ term_t tmp = PL_new_term_ref();
|
|
|
|
PL_put_atom(tmp, ATOM_dot);
|
|
PL_error(NULL, 0, OsError(), ERR_FILE_OPERATION,
|
|
ATOM_getcwd, ATOM_directory, tmp);
|
|
|
|
return NULL;
|
|
}
|
|
|
|
canonisePath(buf);
|
|
GD->paths.CWDlen = strlen(buf);
|
|
buf[GD->paths.CWDlen++] = '/';
|
|
buf[GD->paths.CWDlen] = EOS;
|
|
|
|
if ( GD->paths.CWDdir )
|
|
remove_string(GD->paths.CWDdir);
|
|
GD->paths.CWDdir = store_string(buf);
|
|
}
|
|
|
|
if ( GD->paths.CWDlen < cwdlen )
|
|
{ memcpy(cwd, GD->paths.CWDdir, GD->paths.CWDlen+1);
|
|
return cwd;
|
|
} else
|
|
{ PL_error(NULL, 0, NULL, ERR_REPRESENTATION, ATOM_max_path_length);
|
|
return NULL;
|
|
}
|
|
}
|
|
|
|
|
|
char *
|
|
PL_cwd(char *cwd, size_t cwdlen)
|
|
{ char *rc;
|
|
|
|
LOCK();
|
|
rc = cwd_unlocked(cwd, cwdlen);
|
|
UNLOCK();
|
|
|
|
return rc;
|
|
}
|
|
|
|
|
|
char *
|
|
BaseName(const char *f)
|
|
{ const char *base;
|
|
|
|
for(base = f; *f; f++)
|
|
{ if (*f == '/')
|
|
base = f+1;
|
|
}
|
|
|
|
return (char *)base;
|
|
}
|
|
|
|
|
|
char *
|
|
DirName(const char *f, char *dir)
|
|
{ const char *base, *p;
|
|
|
|
for(base = p = f; *p; p++)
|
|
{ if (*p == '/' && p[1] != EOS )
|
|
base = p;
|
|
}
|
|
if ( base == f )
|
|
{ if ( *f == '/' )
|
|
strcpy(dir, "/");
|
|
else
|
|
strcpy(dir, ".");
|
|
} else
|
|
{ if ( dir != f ) /* otherwise it is in-place */
|
|
strncpy(dir, f, base-f);
|
|
dir[base-f] = EOS;
|
|
}
|
|
|
|
return dir;
|
|
}
|
|
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
bool ChDir(path)
|
|
char *path;
|
|
|
|
Change the current working directory to `path'. File names may depend
|
|
on `path'.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
bool
|
|
ChDir(const char *path)
|
|
{ char ospath[MAXPATHLEN];
|
|
char tmp[MAXPATHLEN];
|
|
|
|
OsPath(path, ospath);
|
|
|
|
if ( path[0] == EOS || streq(path, ".") ||
|
|
(GD->paths.CWDdir && streq(path, GD->paths.CWDdir)) )
|
|
succeed;
|
|
|
|
AbsoluteFile(path, tmp);
|
|
#if __ANDROID__
|
|
/* treat "/assets" as a directory (actually as a mounted file system).
|
|
*
|
|
*/
|
|
if (LOCAL_InAssetDir) {
|
|
free(LOCAL_InAssetDir);
|
|
LOCAL_InAssetDir = NULL;
|
|
}
|
|
if (strstr(ospath, "/assets/") == ospath) {
|
|
const char *dirName = ospath+strlen("/assets/");
|
|
AAssetManager* mgr = GLOBAL_assetManager;
|
|
AAssetDir* dir;
|
|
|
|
if (( dir = AAssetManager_openDir(mgr, dirName) ) &&
|
|
AAssetDir_getNextFileName( dir ) ) {
|
|
// valid directpry
|
|
size_t sz = strlen(ospath)+1;
|
|
AAssetDir_close(dir);
|
|
LOCAL_InAssetDir = (char *)malloc(sz);
|
|
strncpy(LOCAL_InAssetDir, ospath, sz);
|
|
succeed;
|
|
} else {
|
|
fail;
|
|
}
|
|
} else if ( !strcmp(ospath,"/assets") ||
|
|
!strcmp(ospath,"/assets/") ) {
|
|
// valid directpry
|
|
size_t sz = strlen("/assets")+1;
|
|
LOCAL_InAssetDir = (char *)malloc(sz);
|
|
strncpy(LOCAL_InAssetDir, ospath, sz);
|
|
succeed;
|
|
}
|
|
#endif
|
|
|
|
if ( chdir(ospath) == 0 )
|
|
{ size_t len;
|
|
len = strlen(tmp);
|
|
if ( len == 0 || tmp[len-1] != '/' )
|
|
{ tmp[len++] = '/';
|
|
tmp[len] = EOS;
|
|
}
|
|
LOCK(); /* Lock with PL_changed_cwd() */
|
|
GD->paths.CWDlen = len; /* and PL_cwd() */
|
|
if ( GD->paths.CWDdir )
|
|
remove_string(GD->paths.CWDdir);
|
|
GD->paths.CWDdir = store_string(tmp);
|
|
UNLOCK();
|
|
|
|
succeed;
|
|
}
|
|
|
|
fail;
|
|
}
|
|
|
|
|
|
/********************************
|
|
* TIME CONVERSION *
|
|
*********************************/
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
struct tm *PL_localtime_r(time_t time, struct tm *r)
|
|
|
|
Convert tunlime in Unix internal form (seconds since Jan 1 1970) into a
|
|
structure providing easier access to the time.
|
|
|
|
For non-Unix systems: struct time is supposed to look like this.
|
|
Move This definition to pl-os.h and write the conversion functions
|
|
here.
|
|
|
|
struct tm {
|
|
int tm_sec; / * second in the minute (0-59)* /
|
|
int tm_min; / * minute in the hour (0-59) * /
|
|
int tm_hour; / * hour of the day (0-23) * /
|
|
int tm_mday; / * day of the month (1-31) * /
|
|
int tm_mon; / * month of the year (1-12) * /
|
|
int tm_year; / * year (0 = 1900) * /
|
|
int tm_wday; / * day in the week (1-7, 1 = sunday) * /
|
|
int tm_yday; / * day in the year (0-365) * /
|
|
int tm_isdst; / * daylight saving time info * /
|
|
};
|
|
|
|
time_t Time()
|
|
|
|
Return time in seconds after Jan 1 1970 (Unix' time notion).
|
|
|
|
Note: MinGW has localtime_r(), but it is not locked and thus not
|
|
thread-safe. MinGW does not have localtime_s(), but we test for it in
|
|
configure.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
struct tm *
|
|
PL_localtime_r(const time_t *t, struct tm *r)
|
|
{
|
|
#ifdef HAVE_LOCALTIME_R
|
|
return localtime_r(t, r);
|
|
#else
|
|
#ifdef HAVE_LOCALTIME_S
|
|
return localtime_s(r, t) == EINVAL ? NULL : t;
|
|
#else
|
|
struct tm *rc;
|
|
|
|
LOCK();
|
|
if ( (rc = localtime(t)) )
|
|
*r = *rc;
|
|
else
|
|
r = NULL;
|
|
UNLOCK();
|
|
|
|
return r;
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
char *
|
|
PL_asctime_r(const struct tm *tm, char *buf)
|
|
{
|
|
#ifdef HAVE_ASCTIME_R
|
|
return asctime_r(tm, buf);
|
|
#else
|
|
char *rc;
|
|
|
|
LOCK();
|
|
if ( (rc = asctime(tm)) )
|
|
strcpy(buf, rc);
|
|
else
|
|
buf = NULL;
|
|
UNLOCK();
|
|
|
|
return buf;
|
|
#endif
|
|
}
|
|
|
|
|
|
/*******************************
|
|
* TERMINAL *
|
|
*******************************/
|
|
|
|
#ifdef HAVE_TCSETATTR
|
|
#include <termios.h>
|
|
#include <unistd.h>
|
|
#define O_HAVE_TERMIO 1
|
|
#else /*HAVE_TCSETATTR*/
|
|
#ifdef HAVE_SYS_TERMIO_H
|
|
#include <sys/termio.h>
|
|
#define termios termio
|
|
#define O_HAVE_TERMIO 1
|
|
#else
|
|
#ifdef HAVE_SYS_TERMIOS_H
|
|
#include <sys/termios.h>
|
|
#define O_HAVE_TERMIO 1
|
|
#endif
|
|
#endif
|
|
#endif /*HAVE_TCSETATTR*/
|
|
|
|
typedef struct tty_state
|
|
{
|
|
#if defined(O_HAVE_TERMIO)
|
|
struct termios tab;
|
|
#elif defined(HAVE_SGTTYB)
|
|
struct sgttyb tab;
|
|
#else
|
|
int tab; /* empty is not allowed */
|
|
#endif
|
|
} tty_state;
|
|
|
|
#define TTY_STATE(buf) (((tty_state*)(buf->state))->tab)
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
TERMINAL IO MANIPULATION
|
|
|
|
ResetStdin()
|
|
Clear the Sinput buffer after a saved state. Only necessary
|
|
if O_SAVE is defined.
|
|
|
|
PushTty(IOSTREAM *s, ttybuf *buf, int state)
|
|
Push the tty to the specified state and save the old state in
|
|
buf.
|
|
|
|
PopTty(IOSTREAM *s, ttybuf *buf)
|
|
Restore the tty state.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
static void
|
|
ResetStdin(void)
|
|
{ Sinput->limitp = Sinput->bufp = Sinput->buffer;
|
|
if ( !GD->os.org_terminal.read )
|
|
GD->os.org_terminal = *Sinput->functions;
|
|
}
|
|
|
|
static ssize_t
|
|
Sread_terminal(void *handle, char *buf, size_t size)
|
|
{ GET_LD
|
|
intptr_t h = (intptr_t)handle;
|
|
int fd = (int)h;
|
|
source_location oldsrc = LD->read_source;
|
|
|
|
if ( Soutput && True(Soutput, SIO_ISATTY) )
|
|
{ if ( LD->prompt.next && ttymode != TTY_RAW )
|
|
PL_write_prompt(TRUE);
|
|
else
|
|
Sflush(Suser_output);
|
|
}
|
|
|
|
PL_dispatch(fd, PL_DISPATCH_WAIT);
|
|
size = (*GD->os.org_terminal.read)(handle, buf, size);
|
|
|
|
if ( size == 0 ) /* end-of-file */
|
|
{ if ( fd == 0 )
|
|
{ Sclearerr(Suser_input);
|
|
LD->prompt.next = TRUE;
|
|
}
|
|
} else if ( size > 0 && buf[size-1] == '\n' )
|
|
LD->prompt.next = TRUE;
|
|
|
|
LD->read_source = oldsrc;
|
|
|
|
return size;
|
|
}
|
|
|
|
void
|
|
ResetTty(void)
|
|
{ GET_LD
|
|
startCritical;
|
|
ResetStdin();
|
|
|
|
if ( !GD->os.iofunctions.read )
|
|
{ GD->os.iofunctions = *Sinput->functions;
|
|
GD->os.iofunctions.read = Sread_terminal;
|
|
|
|
Sinput->functions =
|
|
Soutput->functions =
|
|
Serror->functions = &GD->os.iofunctions;
|
|
}
|
|
LD->prompt.next = TRUE;
|
|
endCritical;
|
|
}
|
|
|
|
#ifdef O_HAVE_TERMIO /* sys/termios.h or sys/termio.h */
|
|
|
|
#ifndef HAVE_TCSETATTR
|
|
#ifndef NO_SYS_IOCTL_H_WITH_SYS_TERMIOS_H
|
|
#include <sys/ioctl.h>
|
|
#endif
|
|
#ifndef TIOCGETA
|
|
#define TIOCGETA TCGETA
|
|
#endif
|
|
#endif
|
|
|
|
bool
|
|
PushTty(IOSTREAM *s, ttybuf *buf, int mode)
|
|
{ GET_LD
|
|
struct termios tio;
|
|
int fd;
|
|
|
|
buf->mode = ttymode;
|
|
buf->state = NULL;
|
|
ttymode = mode;
|
|
|
|
if ( (fd = Sfileno(s)) < 0 || !isatty(fd) )
|
|
succeed; /* not a terminal */
|
|
if ( !truePrologFlag(PLFLAG_TTY_CONTROL) )
|
|
succeed;
|
|
|
|
buf->state = allocHeapOrHalt(sizeof(tty_state));
|
|
|
|
#ifdef HAVE_TCSETATTR
|
|
if ( tcgetattr(fd, &TTY_STATE(buf)) ) /* save the old one */
|
|
fail;
|
|
#else
|
|
if ( ioctl(fd, TIOCGETA, &TTY_STATE(buf)) ) /* save the old one */
|
|
fail;
|
|
#endif
|
|
|
|
tio = TTY_STATE(buf);
|
|
|
|
switch( mode )
|
|
{ case TTY_RAW:
|
|
#if defined(HAVE_TCSETATTR) && defined(HAVE_CFMAKERAW)
|
|
cfmakeraw(&tio);
|
|
tio.c_oflag = TTY_STATE(buf).c_oflag; /* donot change output modes */
|
|
tio.c_lflag |= ISIG;
|
|
#else
|
|
tio.c_lflag &= ~(ECHO|ICANON);
|
|
#endif
|
|
/* OpenBSD requires this anyhow!? */
|
|
/* Bug in OpenBSD or must we? */
|
|
/* Could this do any harm? */
|
|
tio.c_cc[VTIME] = 0, tio.c_cc[VMIN] = 1;
|
|
break;
|
|
case TTY_OUTPUT:
|
|
tio.c_oflag |= (OPOST|ONLCR);
|
|
break;
|
|
case TTY_SAVE:
|
|
succeed;
|
|
default:
|
|
sysError("Unknown PushTty() mode: %d", mode);
|
|
/*NOTREACHED*/
|
|
}
|
|
|
|
#ifdef HAVE_TCSETATTR
|
|
if ( tcsetattr(fd, TCSANOW, &tio) != 0 )
|
|
{ static int MTOK_warned; /* MT-OK */
|
|
|
|
if ( !MTOK_warned++ )
|
|
warning("Failed to set terminal: %s", OsError());
|
|
}
|
|
#else
|
|
#ifdef TIOCSETAW
|
|
ioctl(fd, TIOCSETAW, &tio);
|
|
#else
|
|
ioctl(fd, TCSETAW, &tio);
|
|
ioctl(fd, TCXONC, (void *)1);
|
|
#endif
|
|
#endif
|
|
|
|
succeed;
|
|
}
|
|
|
|
|
|
bool
|
|
PopTty(IOSTREAM *s, ttybuf *buf, int do_free)
|
|
{ ttymode = buf->mode;
|
|
|
|
if ( buf->state )
|
|
{ int fd = Sfileno(s);
|
|
|
|
if ( fd >= 0 )
|
|
{
|
|
#ifdef HAVE_TCSETATTR
|
|
tcsetattr(fd, TCSANOW, &TTY_STATE(buf));
|
|
#else
|
|
#ifdef TIOCSETA
|
|
ioctl(fd, TIOCSETA, &TTY_STATE(buf));
|
|
#else
|
|
ioctl(fd, TCSETA, &TTY_STATE(buf));
|
|
ioctl(fd, TCXONC, (void *)1);
|
|
#endif
|
|
#endif
|
|
}
|
|
|
|
if ( do_free )
|
|
{ freeHeap(buf->state, sizeof(tty_state));
|
|
buf->state = NULL;
|
|
}
|
|
}
|
|
|
|
succeed;
|
|
}
|
|
|
|
#else /* O_HAVE_TERMIO */
|
|
|
|
#ifdef HAVE_SGTTYB
|
|
|
|
bool
|
|
PushTty(IOSTREAM *s, ttybuf *buf, int mode)
|
|
{ struct sgttyb tio;
|
|
int fd;
|
|
|
|
buf->mode = ttymode;
|
|
buf->state = NULL;
|
|
ttymode = mode;
|
|
|
|
if ( (fd = Sfileno(s)) < 0 || !isatty(fd) )
|
|
succeed; /* not a terminal */
|
|
if ( !truePrologFlag(PLFLAG_TTY_CONTROL) )
|
|
succeed;
|
|
|
|
buf->state = allocHeapOrHalt(sizeof(tty_state));
|
|
|
|
if ( ioctl(fd, TIOCGETP, &TTY_STATE(buf)) ) /* save the old one */
|
|
fail;
|
|
tio = TTY_STATE(buf);
|
|
|
|
switch( mode )
|
|
{ case TTY_RAW:
|
|
tio.sg_flags |= CBREAK;
|
|
tio.sg_flags &= ~ECHO;
|
|
break;
|
|
case TTY_OUTPUT:
|
|
tio.sg_flags |= (CRMOD);
|
|
break;
|
|
case TTY_SAVE:
|
|
succeed;
|
|
default:
|
|
sysError("Unknown PushTty() mode: %d", mode);
|
|
/*NOTREACHED*/
|
|
}
|
|
|
|
ioctl(fd, TIOCSETP, &tio);
|
|
ioctl(fd, TIOCSTART, NULL);
|
|
|
|
succeed;
|
|
}
|
|
|
|
|
|
bool
|
|
PopTty(IOSTREAM *s, ttybuf *buf, int do_free)
|
|
{ ttymode = buf->mode;
|
|
|
|
if ( buf->state )
|
|
{ int fd = Sfileno(s);
|
|
|
|
if ( fd >= 0 )
|
|
{ ioctl(fd, TIOCSETP, &buf->tab);
|
|
ioctl(fd, TIOCSTART, NULL);
|
|
}
|
|
|
|
if ( do_free )
|
|
{ freeHeap(buf->state, sizeof(tty_state));
|
|
buf->state = NULL;
|
|
}
|
|
}
|
|
|
|
succeed;
|
|
}
|
|
|
|
#else /*HAVE_SGTTYB*/
|
|
|
|
bool
|
|
PushTty(IOSTREAM *s, ttybuf *buf, int mode)
|
|
{ buf->mode = ttymode;
|
|
ttymode = mode;
|
|
|
|
succeed;
|
|
}
|
|
|
|
|
|
bool
|
|
PopTty(IOSTREAM *s, ttybuf *buf, int do_free)
|
|
{ GET_LD
|
|
ttymode = buf->mode;
|
|
if ( ttymode != TTY_RAW )
|
|
LD->prompt.next = TRUE;
|
|
|
|
succeed;
|
|
}
|
|
|
|
#endif /*HAVE_SGTTYB*/
|
|
#endif /*O_HAVE_TERMIO*/
|
|
|
|
|
|
/********************************
|
|
* ENVIRONMENT CONTROL *
|
|
*********************************/
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
Simple library to manipulate the OS environment. The modified
|
|
environment will be passed to child processes and the can also be
|
|
requested via getenv/2 from Prolog. Functions
|
|
|
|
int Setenv(name, value)
|
|
char *name, *value;
|
|
|
|
Set the OS environment variable with name `name'. If it exists
|
|
its value is changed, otherwise a new entry in the environment is
|
|
created. The return value is a pointer to the old value, or NULL if
|
|
the variable is new.
|
|
|
|
int Unsetenv(name)
|
|
char *name;
|
|
|
|
Delete a variable from the environment. Return value is the old
|
|
value, or NULL if the variable did not exist.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
size_t
|
|
getenv3(const char *name, char *buf, size_t len)
|
|
{
|
|
#if O_XOS
|
|
return _xos_getenv(name, buf, len);
|
|
#else
|
|
char *s = getenv(name);
|
|
size_t l;
|
|
|
|
if ( s )
|
|
{ if ( (l=strlen(s)) < len )
|
|
memcpy(buf, s, l+1);
|
|
else if ( len > 0 )
|
|
buf[0] = EOS; /* empty string if not fit */
|
|
|
|
return l;
|
|
}
|
|
|
|
return (size_t)-1;
|
|
#endif
|
|
}
|
|
|
|
|
|
char *
|
|
Getenv(const char *name, char *buf, size_t len)
|
|
{ size_t l = getenv3(name, buf, len);
|
|
|
|
if ( l != (size_t)-1 && l < len )
|
|
return buf;
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
#if defined(HAVE_PUTENV) || defined(HAVE_SETENV)
|
|
|
|
int
|
|
Setenv(char *name, char *value)
|
|
{
|
|
#ifdef HAVE_SETENV
|
|
if ( setenv(name, value, TRUE) != 0 )
|
|
return PL_error(NULL, 0, MSG_ERRNO, ERR_SYSCALL, "setenv");
|
|
#else
|
|
char *buf;
|
|
|
|
if ( *name == '\0' || strchr(name, '=') != NULL )
|
|
{ errno = EINVAL;
|
|
return PL_error(NULL, 0, MSG_ERRNO, ERR_SYSCALL, "setenv");
|
|
}
|
|
|
|
buf = alloca(strlen(name) + strlen(value) + 2);
|
|
|
|
if ( buf )
|
|
{ Ssprintf(buf, "%s=%s", name, value);
|
|
|
|
if ( putenv(store_string(buf)) < 0 )
|
|
return PL_error(NULL, 0, MSG_ERRNO, ERR_SYSCALL, "setenv");
|
|
} else
|
|
return PL_error(NULL, 0, NULL, ERR_NOMEM);
|
|
#endif
|
|
succeed;
|
|
}
|
|
|
|
int
|
|
Unsetenv(char *name)
|
|
{
|
|
#ifdef HAVE_UNSETENV
|
|
#ifdef VOID_UNSETENV
|
|
unsetenv(name);
|
|
#else
|
|
if ( unsetenv(name) < 0 )
|
|
return PL_error(NULL, 0, MSG_ERRNO, ERR_SYSCALL, "unsetenv");
|
|
#endif
|
|
|
|
succeed;
|
|
#else
|
|
if ( !getenv(name) )
|
|
succeed;
|
|
|
|
return Setenv(name, "");
|
|
#endif
|
|
}
|
|
|
|
static void
|
|
initEnviron()
|
|
{
|
|
}
|
|
|
|
#else /*HAVE_PUTENV*/
|
|
|
|
extern char **environ; /* Unix predefined environment */
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
Grow the environment array by one and return the (possibly moved) base
|
|
pointer to the new environment.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
forwards char **growEnviron(char**, int);
|
|
forwards char *matchName(char *, char *);
|
|
forwards void setEntry(char **, char *, char *);
|
|
|
|
static char **
|
|
growEnviron(char **e, int amount)
|
|
{ static int filled;
|
|
static int size = -1;
|
|
|
|
if ( amount == 0 ) /* reset after a dump */
|
|
{ size = -1;
|
|
return e;
|
|
}
|
|
|
|
if ( size < 0 )
|
|
{ char **env, **e1, **e2;
|
|
|
|
for(e1=e, filled=0; *e1; e1++, filled++)
|
|
;
|
|
size = ROUND(filled+10+amount, 32);
|
|
env = (char **)PL_malloc(size * sizeof(char *));
|
|
for ( e1=e, e2=env; *e1; *e2++ = *e1++ )
|
|
;
|
|
*e2 = (char *) NULL;
|
|
filled += amount;
|
|
|
|
return env;
|
|
}
|
|
|
|
filled += amount;
|
|
if ( filled + 1 > size )
|
|
{ char **env, **e1, **e2;
|
|
|
|
size += 32;
|
|
env = (char **)PL_realloc(e, size * sizeof(char *));
|
|
for ( e1=e, e2=env; *e1; *e2++ = *e1++ )
|
|
;
|
|
*e2 = (char *) NULL;
|
|
|
|
return env;
|
|
}
|
|
|
|
return e;
|
|
}
|
|
|
|
|
|
static void
|
|
initEnviron(void)
|
|
{ growEnviron(environ, 0);
|
|
}
|
|
|
|
|
|
static char *
|
|
matchName(const char *e, const char *name)
|
|
{ while( *name && *e == *name )
|
|
e++, name++;
|
|
|
|
if ( (*e == '=' || *e == EOS) && *name == EOS )
|
|
return (*e == '=' ? e+1 : e);
|
|
|
|
return (char *) NULL;
|
|
}
|
|
|
|
|
|
static void
|
|
setEntry(char **e, char *name, char *value)
|
|
{ size_t l = strlen(name);
|
|
|
|
*e = PL_malloc_atomic(l + strlen(value) + 2);
|
|
strcpy(*e, name);
|
|
e[0][l++] = '=';
|
|
strcpy(&e[0][l], value);
|
|
}
|
|
|
|
|
|
char *
|
|
Setenv(char *name, char *value)
|
|
{ char **e;
|
|
char *v;
|
|
int n;
|
|
|
|
for(n=0, e=environ; *e; e++, n++)
|
|
{ if ( (v=matchName(*e, name)) != NULL )
|
|
{ if ( !streq(v, value) )
|
|
setEntry(e, name, value);
|
|
return v;
|
|
}
|
|
}
|
|
environ = growEnviron(environ, 1);
|
|
setEntry(&environ[n], name, value);
|
|
environ[n+1] = (char *) NULL;
|
|
|
|
return (char *) NULL;
|
|
}
|
|
|
|
|
|
char *
|
|
Unsetenv(char *name)
|
|
{ char **e;
|
|
char *v;
|
|
int n;
|
|
|
|
for(n=0, e=environ; *e; e++, n++)
|
|
{ if ( (v=matchName(*e, name)) != NULL )
|
|
{ environ = growEnviron(environ, -1);
|
|
e = &environ[n];
|
|
do
|
|
{ e[0] = e[1];
|
|
e++;
|
|
} while(*e);
|
|
|
|
return v;
|
|
}
|
|
}
|
|
|
|
return (char *) NULL;
|
|
}
|
|
|
|
#endif /*HAVE_PUTENV*/
|
|
|
|
/********************************
|
|
* SYSTEM PROCESSES *
|
|
*********************************/
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
int System(command)
|
|
char *command;
|
|
|
|
Invoke a command on the operating system. The return value is the
|
|
exit status of the command. Return value 0 implies succesful
|
|
completion. If you are not running Unix your C-library might provide
|
|
an alternative.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
#ifdef __unix__
|
|
#define SPECIFIC_SYSTEM 1
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
According to the autoconf docs HAVE_SYS_WAIT_H is set if sys/wait.h is
|
|
defined *and* is POSIX.1 compliant, which implies it uses int status
|
|
argument to wait()
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
#ifdef HAVE_SYS_WAIT_H
|
|
#undef UNION_WAIT
|
|
#include <sys/wait.h>
|
|
#define wait_t int
|
|
|
|
#ifndef WEXITSTATUS
|
|
# define WEXITSTATUS(stat_val) ((unsigned)(stat_val) >> 8)
|
|
#endif
|
|
#ifndef WIFEXITED
|
|
# define WIFEXITED(stat_val) (((stat_val) & 255) == 0)
|
|
#endif
|
|
|
|
#else /*HAVE_SYS_WAIT_H*/
|
|
|
|
#ifdef UNION_WAIT /* Old BSD style wait */
|
|
#include <sys/wait.h>
|
|
#define wait_t union wait
|
|
|
|
#ifndef WEXITSTATUS
|
|
#define WEXITSTATUS(s) ((s).w_status)
|
|
#endif
|
|
#ifndef WTERMSIG
|
|
#define WTERMSIG(s) ((s).w_status)
|
|
#endif
|
|
#endif /*UNION_WAIT*/
|
|
|
|
#endif /*HAVE_SYS_WAIT_H*/
|
|
|
|
typedef void (*sigf_t)(int sig);
|
|
|
|
int
|
|
System(char *cmd)
|
|
{ GET_LD
|
|
int pid;
|
|
char *shell = "/bin/sh";
|
|
int rval;
|
|
sigf_t old_int, old_stop;
|
|
|
|
if ( (pid = fork()) == -1 )
|
|
{ return PL_error("shell", 2, OsError(), ERR_SYSCALL, "fork");
|
|
} else if ( pid == 0 ) /* The child */
|
|
{ Setenv("PROLOGCHILD", "yes");
|
|
PL_cleanup_fork();
|
|
execl(shell, BaseName(shell), "-c", cmd, (char *)0);
|
|
fatalError("Failed to execute %s: %s", shell, OsError());
|
|
fail;
|
|
/*NOTREACHED*/
|
|
} else
|
|
{ wait_t status; /* the parent */
|
|
int n;
|
|
|
|
old_int = (sigf_t)signal(SIGINT, SIG_IGN);
|
|
#ifdef SIGTSTP
|
|
old_stop = (sigf_t)signal(SIGTSTP, SIG_DFL);
|
|
#endif /* SIGTSTP */
|
|
|
|
for(;;)
|
|
{
|
|
#ifdef HAVE_WAITPID
|
|
n = waitpid(pid, &status, 0);
|
|
#else
|
|
n = wait(&status);
|
|
#endif
|
|
if ( n == -1 && errno == EINTR )
|
|
continue;
|
|
if ( n != pid )
|
|
continue;
|
|
break;
|
|
}
|
|
|
|
if ( n == -1 )
|
|
{ term_t tmp = PL_new_term_ref();
|
|
|
|
PL_put_atom_chars(tmp, cmd);
|
|
PL_error("shell", 2, MSG_ERRNO, ERR_SHELL_FAILED, tmp);
|
|
|
|
rval = 1;
|
|
} else if (WIFEXITED(status))
|
|
{ rval = WEXITSTATUS(status);
|
|
#ifdef WIFSIGNALED
|
|
} else if (WIFSIGNALED(status))
|
|
{ term_t tmp = PL_new_term_ref();
|
|
int sig = WTERMSIG(status);
|
|
|
|
PL_put_atom_chars(tmp, cmd);
|
|
PL_error("shell", 2, NULL, ERR_SHELL_SIGNALLED, tmp, sig);
|
|
rval = 1;
|
|
#endif
|
|
} else
|
|
{ rval = 1; /* make gcc happy */
|
|
fatalError("Unknown return code from wait(3)");
|
|
/*NOTREACHED*/
|
|
}
|
|
}
|
|
|
|
signal(SIGINT, old_int); /* restore signal handlers */
|
|
#ifdef SIGTSTP
|
|
signal(SIGTSTP, old_stop);
|
|
#endif /* SIGTSTP */
|
|
|
|
return rval;
|
|
}
|
|
#endif /* __unix__ */
|
|
|
|
|
|
#ifdef HAVE_WINEXEC /* Windows 3.1 */
|
|
#define SPECIFIC_SYSTEM 1
|
|
|
|
int
|
|
System(char *command)
|
|
{ char *msg;
|
|
int rval = WinExec(command, SW_SHOWNORMAL);
|
|
|
|
if ( rval < 32 )
|
|
{ switch( rval )
|
|
{ case 0: msg = "Not enough memory"; break;
|
|
case 2: msg = "File not found"; break;
|
|
case 3: msg = "No path"; break;
|
|
case 5: msg = "Unknown error"; break;
|
|
case 6: msg = "Lib requires separate data segment"; break;
|
|
case 8: msg = "Not enough memory"; break;
|
|
case 10: msg = "Incompatible Windows version"; break;
|
|
case 11: msg = "Bad executable file"; break;
|
|
case 12: msg = "Incompatible operating system"; break;
|
|
case 13: msg = "MS-DOS 4.0 executable"; break;
|
|
case 14: msg = "Unknown executable file type"; break;
|
|
case 15: msg = "Real-mode application"; break;
|
|
case 16: msg = "Cannot start multiple copies"; break;
|
|
case 19: msg = "Executable is compressed"; break;
|
|
case 20: msg = "Invalid DLL"; break;
|
|
case 21: msg = "Application is 32-bits"; break;
|
|
default: msg = "Unknown error";
|
|
}
|
|
|
|
warning("Could not start %s: error %d (%s)",
|
|
command, rval, msg);
|
|
return 1;
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
#endif
|
|
|
|
|
|
#ifdef __WINDOWS__
|
|
#define SPECIFIC_SYSTEM 1
|
|
|
|
/* definition in pl-nt.c */
|
|
#endif
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
Nothing special is needed. Just hope the C-library defines system().
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
#ifndef SPECIFIC_SYSTEM
|
|
|
|
int
|
|
System(command)
|
|
char *command;
|
|
{ return system(command);
|
|
}
|
|
|
|
#endif
|
|
|
|
|
|
/* - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - -
|
|
char *findExecutable(char *buf)
|
|
|
|
Return the path name of the executable of SWI-Prolog.
|
|
- - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - */
|
|
|
|
#ifndef __WINDOWS__ /* Win32 version in pl-nt.c */
|
|
static char * Which(const char *program, char *fullname);
|
|
|
|
char *
|
|
findExecutable(const char *av0, char *buffer)
|
|
{ char *file;
|
|
char buf[MAXPATHLEN];
|
|
char tmp[MAXPATHLEN];
|
|
|
|
if ( !av0 || !PrologPath(av0, buf, sizeof(buf)) )
|
|
return NULL;
|
|
file = Which(buf, tmp);
|
|
|
|
#if __unix__ /* argv[0] can be an #! script! */
|
|
if ( file )
|
|
{ int n, fd;
|
|
char buf[MAXPATHLEN];
|
|
|
|
/* Fails if mode is x-only, but */
|
|
/* then it can't be a script! */
|
|
if ( (fd = open(file, O_RDONLY)) < 0 )
|
|
return strcpy(buffer, file);
|
|
|
|
if ( (n=read(fd, buf, sizeof(buf)-1)) > 0 )
|
|
{ close(fd);
|
|
|
|
buf[n] = EOS;
|
|
if ( strncmp(buf, "#!", 2) == 0 )
|
|
{ char *s = &buf[2], *q;
|
|
while(*s && isBlank(*s))
|
|
s++;
|
|
for(q=s; *q && !isBlank(*q); q++)
|
|
;
|
|
*q = EOS;
|
|
|
|
return strcpy(buffer, s);
|
|
}
|
|
}
|
|
|
|
close(fd);
|
|
}
|
|
#endif /*__unix__*/
|
|
|
|
return strcpy(buffer, file ? file : buf);
|
|
}
|
|
|
|
#ifdef __unix__
|
|
static char *
|
|
okToExec(const char *s)
|
|
{ statstruct stbuff;
|
|
|
|
if (statfunc(s, &stbuff) == 0 && /* stat it */
|
|
S_ISREG(stbuff.st_mode) && /* check for file */
|
|
access(s, X_OK) == 0) /* can be executed? */
|
|
return (char *)s;
|
|
else
|
|
return (char *) NULL;
|
|
}
|
|
#define PATHSEP ':'
|
|
#endif /* __unix__ */
|
|
|
|
#if defined(OS2) || defined(__DOS__) || defined(__WINDOWS__)
|
|
#define EXEC_EXTENSIONS { ".exe", ".com", ".bat", ".cmd", NULL }
|
|
#define PATHSEP ';'
|
|
#endif
|
|
|
|
#ifdef EXEC_EXTENSIONS
|
|
|
|
static char *
|
|
okToExec(const char *s)
|
|
{ static char *extensions[] = EXEC_EXTENSIONS;
|
|
static char **ext;
|
|
|
|
DEBUG(2, Sdprintf("Checking %s\n", s));
|
|
for(ext = extensions; *ext; ext++)
|
|
if ( stripostfix(s, *ext) )
|
|
return ExistsFile(s) ? (char *)s : (char *) NULL;
|
|
|
|
for(ext = extensions; *ext; ext++)
|
|
{ static char path[MAXPATHLEN];
|
|
|
|
strcpy(path, s);
|
|
strcat(path, *ext);
|
|
if ( ExistsFile(path) )
|
|
return path;
|
|
}
|
|
|
|
return (char *) NULL;
|
|
}
|
|
#endif /*EXEC_EXTENSIONS*/
|
|
|
|
static char *
|
|
Which(const char *program, char *fullname)
|
|
{ char *path, *dir;
|
|
char *e;
|
|
|
|
if ( IsAbsolutePath(program) ||
|
|
#if OS2 && EMX
|
|
isDriveRelativePath(program) ||
|
|
#endif /* OS2 */
|
|
isRelativePath(program) ||
|
|
strchr(program, '/') )
|
|
{ if ( (e = okToExec(program)) != NULL )
|
|
{ strcpy(fullname, e);
|
|
|
|
return fullname;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#if OS2 && EMX
|
|
if ((e = okToExec(program)) != NULL)
|
|
{
|
|
getcwd(fullname, MAXPATHLEN);
|
|
strcat(fullname, "/");
|
|
strcat(fullname, e);
|
|
return fullname;
|
|
}
|
|
#endif /* OS2 */
|
|
if ((path = getenv("PATH") ) == 0)
|
|
path = DEFAULT_PATH;
|
|
|
|
while(*path)
|
|
{ if ( *path == PATHSEP )
|
|
{ if ( (e = okToExec(program)) )
|
|
return strcpy(fullname, e);
|
|
else
|
|
path++; /* fix by Ron Hess (hess@sco.com) */
|
|
} else
|
|
{ char tmp[MAXPATHLEN];
|
|
|
|
for(dir = fullname; *path && *path != PATHSEP; *dir++ = *path++)
|
|
;
|
|
if (*path)
|
|
path++; /* skip : */
|
|
if ((dir-fullname) + strlen(program)+2 > MAXPATHLEN)
|
|
continue;
|
|
*dir++ = '/';
|
|
strcpy(dir, program);
|
|
if ( (e = okToExec(OsPath(fullname, tmp))) )
|
|
return strcpy(fullname, e);
|
|
}
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
#endif /*__WINDOWS__*/
|
|
|
|
/** int Pause(double time)
|
|
|
|
Suspend execution `time' seconds. Time is given as a floating point
|
|
number, expressing the time to sleep in seconds. Just about every
|
|
platform requires it own implementation. We provide them in the order of
|
|
preference. The implementations differ on their granularity and whether
|
|
or not they can be interrupted savely restarted. The recent POSIX
|
|
nanosleep() is just about the only function that really works well:
|
|
accurate, interruptable and restartable.
|
|
*/
|
|
|
|
#ifdef __WINDOWS__
|
|
#define PAUSE_DONE 1 /* see pl-nt.c */
|
|
#endif
|
|
|
|
#if !defined(PAUSE_DONE) && defined(HAVE_NANOSLEEP)
|
|
#define PAUSE_DONE 1
|
|
|
|
int
|
|
Pause(double t)
|
|
{ struct timespec req;
|
|
int rc;
|
|
|
|
if ( t < 0.0 )
|
|
succeed;
|
|
|
|
req.tv_sec = (time_t) t;
|
|
req.tv_nsec = (long)((t - floor(t)) * 1000000000);
|
|
|
|
for(;;)
|
|
{ rc = nanosleep(&req, &req);
|
|
if ( rc == -1 && errno == EINTR )
|
|
{ if ( PL_handle_signals() < 0 )
|
|
return FALSE;
|
|
} else
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
#endif /*HAVE_NANOSLEEP*/
|
|
|
|
|
|
#if !defined(PAUSE_DONE) && defined(HAVE_USLEEP)
|
|
#define PAUSE_DONE 1
|
|
|
|
int
|
|
Pause(double t)
|
|
{ if ( t <= 0.0 )
|
|
return TRUE;
|
|
|
|
usleep((unsigned long)(t * 1000000.0));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#endif /*HAVE_USLEEP*/
|
|
|
|
|
|
#if !defined(PAUSE_DONE) && defined(HAVE_SELECT)
|
|
#define PAUSE_DONE 1
|
|
|
|
int
|
|
Pause(double time)
|
|
{ struct timeval timeout;
|
|
|
|
if ( time <= 0.0 )
|
|
return;
|
|
|
|
if ( time < 60.0 ) /* select() is expensive. Does it make sense */
|
|
{ timeout.tv_sec = (long)time;
|
|
timeout.tv_usec = (long)(time * 1000000) % 1000000;
|
|
select(32, NULL, NULL, NULL, &timeout);
|
|
|
|
return TRUE;
|
|
} else
|
|
{ int rc;
|
|
int left = (int)(time+0.5);
|
|
|
|
do
|
|
{ rc = sleep(left);
|
|
if ( rc == -1 && errno == EINTR )
|
|
{ if ( PL_handle_signals() < 0 )
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
}
|
|
left -= rc;
|
|
} while ( rc != 0 );
|
|
}
|
|
}
|
|
|
|
#endif /*HAVE_SELECT*/
|
|
|
|
#if !defined(PAUSE_DONE) && defined(HAVE_DOSSLEEP)
|
|
#define PAUSE_DONE 1
|
|
|
|
int /* a millisecond granualrity. */
|
|
Pause(double time) /* the EMX function sleep uses seconds */
|
|
{ if ( time <= 0.0 ) /* the select() trick does not work at all. */
|
|
return TRUE;
|
|
|
|
DosSleep((ULONG)(time * 1000));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#endif /*HAVE_DOSSLEEP*/
|
|
|
|
#if !defined(PAUSE_DONE) && defined(HAVE_SLEEP)
|
|
#define PAUSE_DONE 1
|
|
|
|
int
|
|
Pause(double t)
|
|
{ if ( t <= 0.5 )
|
|
succeed;
|
|
|
|
sleep((int)(t + 0.5));
|
|
|
|
succeed;
|
|
}
|
|
|
|
#endif /*HAVE_SLEEP*/
|
|
|
|
#if !defined(PAUSE_DONE) && defined(HAVE_DELAY)
|
|
#define PAUSE_DONE 1
|
|
|
|
int
|
|
Pause(double t)
|
|
{ delay((int)(t * 1000));
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
#endif /*HAVE_DELAY*/
|
|
|
|
#ifndef PAUSE_DONE
|
|
int
|
|
Pause(double t)
|
|
{ return notImplemented("sleep", 1);
|
|
}
|
|
#endif
|
|
|
|
BeginPredDefs(system)
|
|
PRED_DEF("shell", 2, shell, 0)
|
|
EndPredDefs
|
|
|
|
// @}
|