1906 lines
		
	
	
		
			47 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
			
		
		
	
	
			1906 lines
		
	
	
		
			47 KiB
		
	
	
	
		
			C
		
	
	
	
	
	
| /*************************************************************************
 | |
|  *									 *
 | |
|  *	 YAP Prolog 							 *
 | |
|  *									 *
 | |
|  *	Yap Prolog was developed at NCCUP - Universidade do Porto	 *
 | |
|  *									 *
 | |
|  * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997	 *
 | |
|  *									 *
 | |
|  **************************************************************************
 | |
|  *									 *
 | |
|  * File:		sysbits.c *
 | |
|  * Last rev:	4/03/88							 *
 | |
|  * mods: *
 | |
|  * comments:	very much machine dependent routines			 *
 | |
|  *									 *
 | |
|  *************************************************************************/
 | |
| #ifdef SCCS
 | |
| static char SccsId[] = "%W% %G%";
 | |
| #endif
 | |
| 
 | |
| #include "sysbits.h"
 | |
| 
 | |
| /// File Error Handler
 | |
| static void FileError(yap_error_number type, Term where, const char *format,
 | |
|                       ...) {
 | |
| 
 | |
|   if (trueLocalPrologFlag(FILEERRORS_FLAG)) {
 | |
|     va_list ap;
 | |
| 
 | |
|     va_start(ap, format);
 | |
|     /* now build the error string */
 | |
|     Yap_Error(type, TermNil, format, ap);
 | |
|     va_end(ap);
 | |
|   }
 | |
| }
 | |
| 
 | |
| static Int p_sh(USES_REGS1);
 | |
| static Int p_shell(USES_REGS1);
 | |
| static Int p_system(USES_REGS1);
 | |
| static Int p_mv(USES_REGS1);
 | |
| static Int p_dir_sp(USES_REGS1);
 | |
| static Int p_getenv(USES_REGS1);
 | |
| static Int p_putenv(USES_REGS1);
 | |
| static Term do_glob(const char *spec, bool ok_to);
 | |
| #ifdef MACYAP
 | |
| 
 | |
| /* #define signal	skel_signal */
 | |
| #endif /* MACYAP */
 | |
| static const char *expandVars(const char *spec, char *u);
 | |
| 
 | |
| void exit(int);
 | |
| 
 | |
| #ifdef _WIN32
 | |
| void Yap_WinError(char *yap_error) {
 | |
|   char msg[256];
 | |
|   /* Error, we could not read time */
 | |
|   FormatMessage(FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
 | |
|                 NULL, GetLastError(), MAKELANGID(LANG_NEUTRAL, SUBLANG_DEFAULT),
 | |
|                 msg, 255, NULL);
 | |
|   Yap_Error(SYSTEM_ERROR_OPERATING_SYSTEM, TermNil, "%s at %s", msg, yap_error);
 | |
| }
 | |
| #endif /* __WINDOWS__ */
 | |
| 
 | |
| #define is_valid_env_char(C)                                                   \
 | |
|   (((C) >= 'a' && (C) <= 'z') || ((C) >= 'A' && (C) <= 'Z') || (C) == '_')
 | |
| 
 | |
| /// is_directory: verifies whether an expanded file name
 | |
| /// points at a readable directory
 | |
| bool Yap_isDirectory(const char *FileName) {
 | |
| 
 | |
|   VFS_t *vfs;
 | |
|   if ((vfs = vfs_owner(FileName))) {
 | |
|     return vfs->isdir(vfs, FileName);
 | |
|   }
 | |
| #ifdef _WIN32
 | |
|   DWORD dwAtts = GetFileAttributes(FileName);
 | |
|   if (dwAtts == INVALID_FILE_ATTRIBUTES)
 | |
|     return false;
 | |
|   return (dwAtts & FILE_ATTRIBUTE_DIRECTORY);
 | |
| #elif HAVE_LSTAT
 | |
|   struct stat buf;
 | |
| 
 | |
|   if (lstat(FileName, &buf) == -1) {
 | |
|     /* return an error number */
 | |
|     return false;
 | |
|   }
 | |
|   return S_ISDIR(buf.st_mode);
 | |
| #else
 | |
|   Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
 | |
|             "stat not available in this configuration");
 | |
|   return false;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| Int exists_directory(USES_REGS1) {
 | |
|     int lvl = push_text_stack();
 | |
|     const char *path = Yap_AbsoluteFile(Yap_TextTermToText(Deref(ARG1) PASS_REGS),true);
 | |
|     bool rc = Yap_isDirectory(path);
 | |
|     pop_text_stack(lvl);
 | |
|     return rc;
 | |
| }
 | |
| 
 | |
| bool Yap_Exists(const char *f) {
 | |
|     VFS_t *vfs;
 | |
|     f = Yap_VFAlloc(f);
 | |
|     if ((vfs = vfs_owner(f))) {
 | |
|         return vfs->exists(vfs, f);
 | |
|     }
 | |
| #if _WIN32
 | |
|     if (_access(f, 0) == 0)
 | |
|     return true;
 | |
|   if (errno == EINVAL) {
 | |
|     Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "bad flags to access");
 | |
|   }
 | |
|   return false;
 | |
| #elif HAVE_ACCESS
 | |
|     if (access(f, F_OK) == 0) {
 | |
|         return true;
 | |
|     }
 | |
|     if (errno == EINVAL) {
 | |
|         Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "bad flags to access");
 | |
|     }
 | |
|     return false;
 | |
| #else
 | |
|     Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
 | |
|             "access not available in this configuration");
 | |
|   return false;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| static int dir_separator(int ch) {
 | |
| #ifdef MAC
 | |
|   return (ch == ':');
 | |
| #elif ATARI || _MSC_VER
 | |
|   return (ch == '\\');
 | |
| #elif defined(__MINGW32__) || defined(__CYGWIN__)
 | |
|   return (ch == '\\' || ch == '/');
 | |
| #else
 | |
|   return (ch == '/');
 | |
| #endif
 | |
| }
 | |
| 
 | |
| int Yap_dir_separator(int ch) { return dir_separator(ch); }
 | |
| 
 | |
| #if __WINDOWS__
 | |
| #include <psapi.h>
 | |
| 
 | |
| char *libdir = NULL;
 | |
| #endif
 | |
| 
 | |
| bool Yap_IsAbsolutePath(const char *p0, bool expand) {
 | |
|   // verify first if expansion is needed: ~/ or $HOME/
 | |
|   const char *p = p0;
 | |
|   bool nrc;
 | |
|   if (expand) {
 | |
|     p = expandVars(p0, LOCAL_FileNameBuf);
 | |
|   }
 | |
| #if _WIN32 || __MINGW32__
 | |
|   nrc = !PathIsRelative(p);
 | |
| #else
 | |
|   nrc = (p[0] == '/');
 | |
| #endif
 | |
|   return nrc;
 | |
| }
 | |
| 
 | |
| #define isValidEnvChar(C)                                                      \
 | |
|   (((C) >= 'a' && (C) <= 'z') || ((C) >= 'A' && (C) <= 'Z') || (C) == '_')
 | |
| 
 | |
| // this is necessary because
 | |
| // support for ~expansion at the beginning
 | |
| // systems like Android do not do this.
 | |
| static const char *PlExpandVars(const char *source, const char *root) {
 | |
|   CACHE_REGS
 | |
|   int lvl = push_text_stack();
 | |
|   const char *src = source;
 | |
|   char *result = Malloc(YAP_FILENAME_MAX + 1);
 | |
| 
 | |
|   if (strlen(source) >= YAP_FILENAME_MAX) {
 | |
|     Yap_Error(SYSTEM_ERROR_OPERATING_SYSTEM, TermNil,
 | |
|               "%s in true_file-name is larger than the buffer size (%d bytes)",
 | |
|               source, strlen(source));
 | |
|   }
 | |
|   /* step 1: eating home information */
 | |
|   if (source[0] == '~') {
 | |
|     if (dir_separator(source[1]) || source[1] == '\0') {
 | |
|       char *s;
 | |
|       src++;
 | |
| #if defined(_WIN32)
 | |
|       s = getenv("HOMEDRIVE");
 | |
|       if (s != NULL)
 | |
|         strncpy(result, getenv("HOMEDRIVE"), YAP_FILENAME_MAX);
 | |
| // s = getenv("HOMEPATH");
 | |
| #else
 | |
|       s = getenv("HOME");
 | |
| #endif
 | |
|       if (s != NULL)
 | |
|         strncpy(result, s, YAP_FILENAME_MAX);
 | |
|       strcat(result, src);
 | |
|       result = pop_output_text_stack(lvl, result);
 | |
|       return result;
 | |
|     } else {
 | |
| #if HAVE_GETPWNAM
 | |
|       struct passwd *user_passwd;
 | |
|       char *res = result;
 | |
| 
 | |
|       src++;
 | |
|       while (!dir_separator((*res = *src)) && *res != '\0')
 | |
|         res++, src++;
 | |
|       res[0] = '\0';
 | |
|       if ((user_passwd = getpwnam(result)) == NULL) {
 | |
|         FileError(SYSTEM_ERROR_OPERATING_SYSTEM,
 | |
|                   MkAtomTerm(Yap_LookupAtom(source)),
 | |
|                   "User %s does not exist in %s", result, source);
 | |
|         pop_text_stack(lvl);
 | |
|         return NULL;
 | |
|       }
 | |
|       strncpy(result, user_passwd->pw_dir, YAP_FILENAME_MAX);
 | |
|       strcat(result, src);
 | |
| #else
 | |
|       FileError(
 | |
|           SYSTEM_ERROR_OPERATING_SYSTEM, MkAtomTerm(Yap_LookupAtom(source)),
 | |
|           "User %s cannot be found in %s, missing getpwnam", result, source);
 | |
|       return NULL;
 | |
| #endif
 | |
|     }
 | |
|     result = pop_output_text_stack(lvl, result);
 | |
|     return result;
 | |
|   }
 | |
|   // do VARIABLE expansion
 | |
|   else if (source[0] == '$') {
 | |
|     /* follow SICStus expansion rules */
 | |
|     char v[YAP_FILENAME_MAX + 1];
 | |
|     int ch;
 | |
|     char *s, *res;
 | |
|     src = source + 1;
 | |
|     if (src[0] == '{') {
 | |
|       res = v;
 | |
|       src++;
 | |
|       while ((*res = (ch = *src++)) && isValidEnvChar(ch) && ch != '}') {
 | |
|         res++;
 | |
|       }
 | |
|       if (ch == '}') {
 | |
|         // {...}
 | |
|         // done
 | |
|         res[0] = '\0';
 | |
|       }
 | |
|     } else {
 | |
|       res = v;
 | |
|       while ((*res = (ch = *src++)) && isValidEnvChar(ch) && ch != '}') {
 | |
|         res++;
 | |
|       }
 | |
|       src--;
 | |
|       res[0] = '\0';
 | |
|     }
 | |
|     if ((s = (char *)getenv(v))) {
 | |
|       strcpy(result, s);
 | |
|       strcat(result, src);
 | |
|     } else
 | |
|       strcpy(result, src);
 | |
|   } else {
 | |
|     size_t tocp = strlen(src);
 | |
|     if (root) {
 | |
|       tocp = strlen(root) + 1;
 | |
|     }
 | |
|     if (tocp > YAP_FILENAME_MAX) {
 | |
|       Yap_Error(SYSTEM_ERROR_OPERATING_SYSTEM, MkStringTerm(src),
 | |
|                 "path too long");
 | |
|       pop_text_stack(lvl);
 | |
|       return NULL;
 | |
|     }
 | |
|     if (root && !Yap_IsAbsolutePath(source, false)) {
 | |
|       strncpy(result, root, YAP_FILENAME_MAX);
 | |
|       if (root[strlen(root) - 1] != '/')
 | |
|         strncat(result, "/", YAP_FILENAME_MAX);
 | |
|       strncat(result, source, YAP_FILENAME_MAX);
 | |
|     } else {
 | |
|       strncpy(result, source, strlen(src) + 1);
 | |
|     }
 | |
|   }
 | |
|   result = pop_output_text_stack(lvl, result);
 | |
|   return result;
 | |
| }
 | |
| 
 | |
| #if _WIN32
 | |
| // straightforward conversion from Unix style to WIN style
 | |
| // check cygwin path.cc for possible improvements
 | |
| static char *unix2win(const char *source, char *target, int max) {
 | |
|   char *s = target;
 | |
|   const char *s0 = source;
 | |
|   char *s1;
 | |
|   int ch;
 | |
| 
 | |
|   if (s == NULL)
 | |
|     s = malloc(YAP_FILENAME_MAX + 1);
 | |
|   s1 = s;
 | |
|   // win32 syntax
 | |
|   // handle drive notation, eg //a/
 | |
|   if (s0[0] == '\0') {
 | |
|     s[0] = '.';
 | |
|     s[1] = '\0';
 | |
|     return s;
 | |
|   }
 | |
|   if (s0[0] == '/' && s0[1] == '/' && isalpha(s0[2]) && s0[3] == '/') {
 | |
|     s1[0] = s0[2];
 | |
|     s1[1] = ':';
 | |
|     s1[2] = '\\';
 | |
|     s0 += 4;
 | |
|     s1 += 3;
 | |
|   }
 | |
|   while ((ch = *s1++ = *s0++)) {
 | |
|     if (ch == '$') {
 | |
|       s1[-1] = '%';
 | |
|       ch = *s0;
 | |
|       // handle $(....)
 | |
|       if (ch == '{') {
 | |
|         s0++;
 | |
|         while ((ch = *s0++) != '}') {
 | |
|           *s1++ = ch;
 | |
|           if (ch == '\0')
 | |
|             return FALSE;
 | |
|         }
 | |
|         *s1++ = '%';
 | |
|       } else {
 | |
|         while (((ch = *s1++ = *s0++) >= 'A' && ch <= 'Z') ||
 | |
|                (ch >= 'a' && ch <= 'z') || (ch == '-') ||
 | |
|                (ch >= '0' && ch <= '9') || (ch == '_'))
 | |
|           ;
 | |
|         s1[-1] = '%';
 | |
|         *s1++ = ch;
 | |
|         if (ch == '\0') {
 | |
|           s1--;
 | |
|           s0--;
 | |
|         }
 | |
|       }
 | |
|     } else if (ch == '/')
 | |
|       s1[-1] = '\\';
 | |
|   }
 | |
|   return s;
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static char *OsPath(const char *p, char *buf) { return (char *)p; }
 | |
| 
 | |
| static char *PrologPath(const char *Y, char *X) { return (char *)Y; }
 | |
| 
 | |
| #if _WIN32
 | |
| #define HAVE_BASENAME 1
 | |
| #define HAVE_REALPATH 1
 | |
| #endif
 | |
| 
 | |
| extern char *virtual_cwd;
 | |
| 
 | |
| bool Yap_ChDir(const char *path) {
 | |
|   bool rc = false;
 | |
|   int lvl = push_text_stack();
 | |
| 
 | |
|     const char *qpath = Yap_AbsoluteFile(path, true);
 | |
|   //__android_log_print(ANDROID_LOG_INFO, "YAPDroid", "chdir %s", path);
 | |
|   VFS_t *v;
 | |
|   if ((v = vfs_owner(qpath))) {
 | |
|     rc = v->chdir(v, (qpath));
 | |
|     pop_text_stack(lvl);
 | |
|     return rc;
 | |
|   }
 | |
| #if _WIN32
 | |
|   rc = true;
 | |
|   if (qpath != NULL && qpath[0] &&
 | |
|       (rc = (SetCurrentDirectory(qpath) != 0)) == 0) {
 | |
|     Yap_WinError("SetCurrentDirectory failed");
 | |
|   }
 | |
| #else
 | |
|   rc = (chdir(qpath) == 0);
 | |
| #endif
 | |
|   pop_text_stack(lvl);
 | |
|   return rc;
 | |
| }
 | |
| 
 | |
| static char *close_path(char *b0, char *o0, char *o) {
 | |
| 
 | |
|   if (b0[0] == '\0') {
 | |
|     return o;
 | |
|   } else if (!strcmp(b0, "..")) {
 | |
|     while (o-- > o0) {
 | |
|       if (dir_separator(*o)) {
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|   } else if (strcmp(b0, ".") != 0) {
 | |
|     *o++ = '/';
 | |
|     strcpy(o, b0);
 | |
|     o += strlen(b0);
 | |
|   }
 | |
|   return o;
 | |
| }
 | |
| 
 | |
| static char *clean_path(const char *path) {
 | |
|   const char *p, *p0;
 | |
|   int lvl = push_text_stack();
 | |
| 
 | |
|   //__android_log_print(ANDROID_LOG_INFO, "YAPDroid ", " looking at %s", path);
 | |
|   char *o0 = Malloc(FILENAME_MAX + 1), *o = o0;
 | |
|   int ch;
 | |
|   char *b0 = Malloc(FILENAME_MAX + 1), *b = b0;
 | |
|   p = p0 = path;
 | |
|   while ((ch = *p++)) {
 | |
|     if (dir_separator(ch)) {
 | |
|       if (b == b0) {
 | |
|         o = o0;
 | |
|       } else {
 | |
|         b[0] = '\0';
 | |
|         o = close_path(b0, o0, o);
 | |
|         b = b0;
 | |
|       }
 | |
|     } else {
 | |
|       *b++ = ch;
 | |
|     }
 | |
|   }
 | |
|   if (!dir_separator(p[-1])) {
 | |
|     b[0] = '\0';
 | |
|     o = close_path(b0, o0, o);
 | |
|   }
 | |
|   if (o == o0)
 | |
|     *o++ = '/';
 | |
|   *o = '\0';
 | |
| //  __android_log_print(ANDROID_LOG_INFO, "YAPDroid ", " %s at %s, %p-%p", p0, o0,
 | |
| //                      o, o0);
 | |
|   return pop_output_text_stack(lvl, o0);
 | |
| }
 | |
| 
 | |
| static const char *myrealpath(const char *path USES_REGS) {
 | |
| 
 | |
|   int lvl = push_text_stack();
 | |
|   VFS_t *v;
 | |
|   char *out, *o;
 | |
|   if (Yap_IsAbsolutePath(path, true)) {
 | |
|     o = clean_path(path);
 | |
|     return pop_output_text_stack(lvl, o);
 | |
| 
 | |
|   } else {
 | |
|     out = Malloc(FILENAME_MAX + 1);
 | |
|     Yap_getcwd(out, FILENAME_MAX);
 | |
|     strcat(out, "/");
 | |
|     strcat(out, path);
 | |
|     o = clean_path(out);
 | |
|     if ((v = vfs_owner(o))) {
 | |
|       return pop_output_text_stack(lvl, o);
 | |
|     }
 | |
|   }
 | |
| #if _WIN32
 | |
|   DWORD retval = 0;
 | |
| 
 | |
|   // notice that the file does not need to exist
 | |
|   retval = GetFullPathName(path, YAP_FILENAME_MAX, o, NULL);
 | |
|   if (retval == 0) {
 | |
|     pop_text_stack(lvl);
 | |
|     Yap_WinError("Generating a full path name for a file");
 | |
|     return NULL;
 | |
|   }
 | |
|   return pop_output_text_stack(lvl, o);
 | |
| #elif HAVE_REALPATH
 | |
|   {
 | |
|     char *rc = realpath(path, o);
 | |
| 
 | |
|     if (rc) {
 | |
|       return pop_output_text_stack(lvl, rc);
 | |
|     }
 | |
|     // rc = NULL;
 | |
|     if (errno == ENOENT || errno == EACCES) {
 | |
|       char *base = Malloc(FILENAME_MAX + 1);
 | |
|       strncpy(base, path, FILENAME_MAX);
 | |
|       char *p = base + strlen(base);
 | |
|       while (p > base && !dir_separator(*--p))
 | |
|         ;
 | |
|       if (p == base)
 | |
|         p[1] = '\0';
 | |
|       else
 | |
|         p[0] = '\0';
 | |
|       char *tmp = Malloc(FILENAME_MAX + 1);
 | |
|       rc = realpath(base, tmp);
 | |
| 
 | |
|       if (rc) {
 | |
|         // base may have been destroyed
 | |
|         char *b = base + strlen(base);
 | |
|         while (b > base && !dir_separator(*--b))
 | |
|           ;
 | |
|         if (b[0] && !dir_separator(b[0]))
 | |
|           b++;
 | |
|         size_t e = strlen(rc);
 | |
|         size_t bs = strlen(b);
 | |
| 
 | |
|         if (rc != out && rc != base) {
 | |
|           rc = Realloc(rc, e + bs + 2);
 | |
|         }
 | |
| #if _WIN32
 | |
|         if (rc[e - 1] != '\\' && rc[e - 1] != '/') {
 | |
|           rc[e] = '\\';
 | |
|           rc[e + 1] = '\0';
 | |
|         }
 | |
| #else
 | |
|         if (rc[e - 1] != '/') {
 | |
|           rc[e] = '/';
 | |
|           rc[e + 1] = '\0';
 | |
|         }
 | |
| #endif
 | |
|         strcat(rc, b);
 | |
|         rc = pop_output_text_stack(lvl, rc);
 | |
|         return rc;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| #endif
 | |
|   pop_text_stack(lvl);
 | |
|   return path;
 | |
| }
 | |
| 
 | |
| static const char *expandVars(const char *spec, char *u) {
 | |
|   CACHE_REGS
 | |
| #if _WIN32 || defined(__MINGW32__)
 | |
|   char *out;
 | |
| 
 | |
|   // first pass, remove Unix style stuff
 | |
|   if (spec == NULL || (out = unix2win(spec, u, YAP_FILENAME_MAX)) == NULL)
 | |
|     return NULL;
 | |
|   spec = u;
 | |
| #endif
 | |
|   bool ok_to = LOCAL_PrologMode && !(LOCAL_PrologMode & BootMode);
 | |
|   if (ok_to) {
 | |
|     Term t = do_glob(spec, true);
 | |
|     if (IsPairTerm(t))
 | |
|       return RepAtom(AtomOfTerm(HeadOfTerm(t)))->StrOfAE;
 | |
|     return NULL;
 | |
|   }
 | |
|   return spec;
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * generate absolute path, if ok first expand SICStus Prolog style
 | |
|  *
 | |
|  * @param[in]  spec the file path, including `~` and `$`.
 | |
|  * @param[in]  ok where to process `~` and `$`.
 | |
|  *
 | |
|  * @return tmp, or NULL, in malloced memory
 | |
|  */
 | |
| const char *Yap_AbsoluteFile(const char *spec, bool ok) {
 | |
|   const char *rc;
 | |
|   const char *spec1;
 | |
|   const char *spec2;
 | |
|   int lvl = push_text_stack();
 | |
| 
 | |
|   /// spec gothe original spec;
 | |
|   /// rc0 may be an outout buffer
 | |
|   /// rc1 the internal buffer
 | |
|   ///
 | |
|   /// PlExpandVars
 | |
| 
 | |
| #if _WIN32
 | |
|   char rc2[YAP_FILENAME_MAX];
 | |
|   if ((rc = unix2win(spec, rc2, YAP_FILENAME_MAX)) == NULL) {
 | |
|     return NULL;
 | |
|   }
 | |
|   spec1 = rc;
 | |
| #else
 | |
|   spec1 = spec;
 | |
| #endif
 | |
|   /// spec gothe original spec;
 | |
|   /// rc1 the internal buffer
 | |
|   if (ok) {
 | |
|     const char *q = PlExpandVars(spec1, NULL);
 | |
|     if (!q)
 | |
|       spec2 = spec1;
 | |
|     else
 | |
|       spec2 = q;
 | |
|   } else {
 | |
|     spec2 = spec1;
 | |
|   }
 | |
|   rc = myrealpath(spec2 PASS_REGS);
 | |
|   return pop_output_text_stack(lvl, rc);
 | |
| }
 | |
| 
 | |
| static Term
 | |
| /* Expand the string for the program to run.  */
 | |
| do_glob(const char *spec, bool glob_vs_wordexp) {
 | |
|   CACHE_REGS
 | |
|   if (spec == NULL) {
 | |
|     return TermNil;
 | |
|   }
 | |
| #if _WIN32
 | |
|   char u[YAP_FILENAME_MAX + 1];
 | |
|   {
 | |
|     WIN32_FIND_DATA find;
 | |
|     HANDLE hFind;
 | |
|     CELL *dest;
 | |
|     Term tf;
 | |
|     char drive[_MAX_DRIVE];
 | |
|     char dir[_MAX_DIR];
 | |
|     char fname[_MAX_FNAME];
 | |
|     char ext[_MAX_EXT];
 | |
| 
 | |
|     _splitpath(spec, drive, dir, fname, ext);
 | |
|     _makepath(u, drive, dir, fname, ext);
 | |
| 
 | |
|     // first pass, remove Unix style stuff
 | |
|     hFind = FindFirstFile(u, &find);
 | |
|     if (hFind == INVALID_HANDLE_VALUE) {
 | |
|       return TermNil;
 | |
|     } else {
 | |
|       tf = AbsPair(HR);
 | |
|       _makepath(u, drive, dir, find.cFileName, NULL);
 | |
|       HR[0] = MkAtomTerm(Yap_LookupAtom(u));
 | |
|       HR[1] = TermNil;
 | |
|       dest = HR + 1;
 | |
|       HR += 2;
 | |
|       while (FindNextFile(hFind, &find)) {
 | |
|         *dest = AbsPair(HR);
 | |
|         _makepath(u, drive, dir, find.cFileName, NULL);
 | |
|         HR[0] = MkAtomTerm(Yap_LookupAtom(u));
 | |
|         HR[1] = TermNil;
 | |
|         dest = HR + 1;
 | |
|         HR += 2;
 | |
|       }
 | |
|       FindClose(hFind);
 | |
|     }
 | |
|     return tf;
 | |
|   }
 | |
| #elif __ANDROID__
 | |
|      return MkPairTerm(MkAtomTerm(Yap_LookupAtom(spec)), TermNil);
 | |
| #elif HAVE_WORDEXP || HAVE_GLOB
 | |
|   char u[YAP_FILENAME_MAX + 1];
 | |
|   const char *espec = u;
 | |
|   strncpy(u, spec, sizeof(u));
 | |
|   /* Expand the string for the program to run.  */
 | |
|   size_t pathcount;
 | |
| #if HAVE_GLOB
 | |
|   glob_t gresult;
 | |
| #endif
 | |
| #if HAVE_WORDEXP
 | |
|   wordexp_t wresult;
 | |
| #endif
 | |
| #if HAVE_GLOB || HAVE_WORDEXP
 | |
|   char **ss = NULL;
 | |
|   int flags = 0, j;
 | |
| #endif
 | |
|   if (glob_vs_wordexp) {
 | |
| #if HAVE_GLOB
 | |
| #ifdef GLOB_NOCHECK
 | |
|     flags = GLOB_NOCHECK;
 | |
| #else
 | |
|     flags = 0;
 | |
| #endif
 | |
| #ifdef GLOB_BRACE
 | |
|     flags |= GLOB_BRACE | GLOB_TILDE;
 | |
| #endif
 | |
|     switch (glob(espec, flags, NULL, &gresult)) {
 | |
|     case 0: /* Successful.  */
 | |
|       ss = gresult.gl_pathv;
 | |
|       pathcount = gresult.gl_pathc;
 | |
|       if (pathcount) {
 | |
|         break;
 | |
|       }
 | |
|     case GLOB_NOMATCH:
 | |
|       globfree(&gresult);
 | |
|       { return TermNil; }
 | |
|     case GLOB_ABORTED:
 | |
|       PlIOError(SYSTEM_ERROR_OPERATING_SYSTEM, ARG1, "glob aborted: %sn",
 | |
|                 strerror(errno));
 | |
|       globfree(&gresult);
 | |
|       return TermNil;
 | |
|     case GLOB_NOSPACE:
 | |
|       Yap_Error(RESOURCE_ERROR_HEAP, ARG1, "glob ran out of space: %sn",
 | |
|                 strerror(errno));
 | |
|       globfree(&gresult);
 | |
|       return TermNil;
 | |
|     /* If the error was WRDE_NOSPACE,
 | |
|      then perhaps part of the result was allocated.  */
 | |
|     default: /* Some other error.  */
 | |
|       return TermNil;
 | |
|     }
 | |
| #endif
 | |
|   } else {
 | |
| #if HAVE_WORDEXP
 | |
|     int rc;
 | |
|     memset(&wresult, 0, sizeof(wresult));
 | |
|     switch ((rc = wordexp(espec, &wresult, flags))) {
 | |
|     case 0: /* Successful.  */
 | |
|       ss = wresult.we_wordv;
 | |
|       pathcount = wresult.we_wordc;
 | |
|       if (pathcount) {
 | |
|         break;
 | |
|       } else {
 | |
|         Term t;
 | |
|         t = MkAtomTerm(Yap_LookupAtom(expandVars(espec, NULL)));
 | |
|         wordfree(&wresult);
 | |
|         return MkPairTerm(t, TermNil);
 | |
|       }
 | |
|     case WRDE_NOSPACE:
 | |
|       /* If the error was WRDE_NOSPACE,
 | |
|        then perhaps part of the result was allocated.  */
 | |
|       Yap_Error(RESOURCE_ERROR_HEAP, ARG1, "wordexp ran out of space: %s",
 | |
|                 strerror(errno));
 | |
|       wordfree(&wresult);
 | |
|       return TermNil;
 | |
|     default: /* Some other error.  */
 | |
|       PlIOError(SYSTEM_ERROR_OPERATING_SYSTEM, ARG1, "wordexp failed: %s",
 | |
|                 strerror(errno));
 | |
|       wordfree(&wresult);
 | |
|       return TermNil;
 | |
|     }
 | |
| #endif
 | |
|   }
 | |
|   const char *tmp;
 | |
|   Term tf = TermNil;
 | |
|   for (j = 0; j < pathcount; j++) {
 | |
|     const char *s = ss[pathcount - (j + 1)];
 | |
|     tmp = s;
 | |
|     // if (!exists(s))
 | |
|     //  continue;
 | |
|     Atom a = Yap_LookupAtom(tmp);
 | |
|     tf = MkPairTerm(MkAtomTerm(a), tf);
 | |
|   }
 | |
| #if HAVE_GLOB
 | |
|   if (glob_vs_wordexp)
 | |
|     globfree(&gresult);
 | |
| #endif
 | |
| #if HAVE_WORDEXP
 | |
|   if (!glob_vs_wordexp)
 | |
|     wordfree(&wresult);
 | |
| #endif
 | |
|   return tf;
 | |
| #else
 | |
|   // just use basic
 | |
|   return MkPairTerm(MkAtomTerm(Yap_LookupAtom(spec)), TermNil);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /**
 | |
|  * @pred prolog_expanded_file_system_path( +PrologPath, +ExpandVars, -OSPath )
 | |
|  *
 | |
|  * Apply basic transformations to paths, and conidtionally apply
 | |
|  * traditional SICStus-style variable expansion.
 | |
|  *
 | |
|  * @param PrologPath the source, may be atom or string
 | |
|  * @param ExpandVars expand initial occurrence of ~ or $
 | |
|  * @param Prefix add this path before _PrologPath_
 | |
|  * @param OSPath pathname.
 | |
|  *
 | |
|  * @return
 | |
|  */
 | |
| static Int real_path(USES_REGS1) {
 | |
|   Term t1 = Deref(ARG1);
 | |
|   const char *cmd, *rc0;
 | |
| 
 | |
|   if (IsAtomTerm(t1)) {
 | |
|     cmd = RepAtom(AtomOfTerm(t1))->StrOfAE;
 | |
|   } else if (IsStringTerm(t1)) {
 | |
|     cmd = StringOfTerm(t1);
 | |
|   } else {
 | |
|     return false;
 | |
|   }
 | |
| #if _WIN32
 | |
|   char cmd2[YAP_FILENAME_MAX + 1];
 | |
|   char *rc;
 | |
| 
 | |
|   if ((rc = unix2win(cmd, cmd2, YAP_FILENAME_MAX)) == NULL) {
 | |
|     return false;
 | |
|   }
 | |
|   cmd = rc;
 | |
| #endif
 | |
|   int lvl = push_text_stack();
 | |
|   rc0 = myrealpath(cmd PASS_REGS);
 | |
|   if (!rc0) {
 | |
|     PlIOError(SYSTEM_ERROR_OPERATING_SYSTEM, ARG1, NULL);
 | |
|   }
 | |
|   bool out = Yap_unify(MkAtomTerm(Yap_LookupAtom(rc0)), ARG2);
 | |
|   pop_output_text_stack(lvl, rc0);
 | |
| 
 | |
|   return out;
 | |
| }
 | |
| 
 | |
| #define EXPAND_FILENAME_DEFS()                                                 \
 | |
|   PAR("parameter_expansion", isatom, EXPAND_FILENAME_PARAMETER_EXPANSION)      \
 | |
|   , PAR("commands", booleanFlag, EXPAND_FILENAME_COMMANDS),                    \
 | |
|       PAR(NULL, ok, EXPAND_FILENAME_END)
 | |
| 
 | |
| #define PAR(x, y, z) z
 | |
| 
 | |
| typedef enum expand_filename_enum_choices {
 | |
|   EXPAND_FILENAME_DEFS()
 | |
| } expand_filename_enum_choices_t;
 | |
| 
 | |
| #undef PAR
 | |
| 
 | |
| #define PAR(x, y, z)                                                           \
 | |
|   { x, y, z }
 | |
| 
 | |
| static const param_t expand_filename_defs[] = {EXPAND_FILENAME_DEFS()};
 | |
| #undef PAR
 | |
| 
 | |
| static Term do_expand_file_name(Term t1, Term opts USES_REGS) {
 | |
|   xarg *args;
 | |
|   expand_filename_enum_choices_t i;
 | |
|   bool use_system_expansion = true;
 | |
|   const char *tmpe = NULL;
 | |
|   const char *spec;
 | |
|   Term tf;
 | |
| 
 | |
|   if (IsVarTerm(t1)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, t1, NULL);
 | |
|     return TermNil;
 | |
|   } else if (IsAtomTerm(t1)) {
 | |
|     spec = AtomTermName(t1);
 | |
|   } else if (IsStringTerm(t1)) {
 | |
|     spec = StringOfTerm(t1);
 | |
|   } else {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, t1, NULL);
 | |
|     return TermNil;
 | |
|   }
 | |
| #if _WIN32
 | |
|   char *rc;
 | |
|   char cmd2[YAP_FILENAME_MAX + 1];
 | |
| 
 | |
|   if ((rc = unix2win(spec, cmd2, YAP_FILENAME_MAX)) == NULL) {
 | |
|     return false;
 | |
|   }
 | |
|   spec = rc;
 | |
| #endif
 | |
| 
 | |
|   args = Yap_ArgListToVector(
 | |
|       opts, expand_filename_defs,
 | |
|       EXPAND_FILENAME_END,DOMAIN_ERROR_EXPAND_FILENAME_OPTION);
 | |
|   if (args == NULL) {
 | |
|     return TermNil;
 | |
|   }
 | |
|   tmpe = NULL;
 | |
| 
 | |
|   for (i = 0; i < EXPAND_FILENAME_END; i++) {
 | |
|     if (args[i].used) {
 | |
|       Term t = args[i].tvalue;
 | |
|       switch (i) {
 | |
|       case EXPAND_FILENAME_PARAMETER_EXPANSION:
 | |
|         if (t == TermProlog) {
 | |
|           tmpe = expandVars(spec, LOCAL_FileNameBuf);
 | |
|           if (tmpe == NULL) {
 | |
|             free(args);
 | |
|             return TermNil;
 | |
|           }
 | |
|           spec = tmpe;
 | |
|         } else if (t == TermTrue) {
 | |
|           use_system_expansion = true;
 | |
|         } else if (t == TermFalse) {
 | |
|           use_system_expansion = false;
 | |
|         }
 | |
|         break;
 | |
|       case EXPAND_FILENAME_COMMANDS:
 | |
|         if (!use_system_expansion) {
 | |
|           use_system_expansion = true;
 | |
| #if 0 // def WRDE_NOCMD
 | |
|                     if (t == TermFalse) {
 | |
|                         flags = WRDE_NOCMD;
 | |
|                     }
 | |
| #endif
 | |
|         }
 | |
|       case EXPAND_FILENAME_END:
 | |
|         break;
 | |
|       }
 | |
|     }
 | |
|   }
 | |
|   free(args);
 | |
|   if (!use_system_expansion) {
 | |
|     const char *o = expandVars(spec, NULL);
 | |
|     if (!o)
 | |
|       return false;
 | |
|     tf = MkPairTerm(MkAtomTerm(Yap_LookupAtom(o)), TermNil);
 | |
| #if _WIN32
 | |
|     if (o != cmd2)
 | |
| #endif
 | |
|     {
 | |
|       freeBuffer(o);
 | |
|     }
 | |
|   } else {
 | |
|     tf = do_glob(spec, true);
 | |
|   }
 | |
|   if (tmpe
 | |
| #if _WIN32
 | |
|       && tmpe != cmd2
 | |
| #endif
 | |
|   ) {
 | |
|     freeBuffer(tmpe);
 | |
|   }
 | |
|   return tf;
 | |
| }
 | |
| 
 | |
| /* @pred expand_file_name( +Pattern, -ListOfPaths) is det
 | |
| 
 | |
| This builtin receives a pattern and expands it into a list of files.
 | |
|   In Unix-like systems, YAP applies glob to expand patterns such as '*', '.',
 | |
| and '?'. Further variable expansion
 | |
| may also happen. glob is shell-dependent: som   Yap_InitCPred
 | |
| ("absolute_file_system_path", 2, absolute_file_system_path, 0);
 | |
|     Yap_InitCPred ("real_path", 2, prolog_realpath, 0);
 | |
|     Yap_InitCPred ("true_file_name", 2,
 | |
|                    true_file_name, SyncPredFlag);
 | |
|     Yap_InitCPred ("true_file_name", 3, true_file_name3, SyncPredFlag);
 | |
| e shells allow command execution and brace-expansion.
 | |
| 
 | |
| */
 | |
| static Int expand_file_name(USES_REGS1) {
 | |
|   Term tf = do_expand_file_name(Deref(ARG1), TermNil PASS_REGS);
 | |
|   if (tf == 0)
 | |
|     return false;
 | |
|   return Yap_unify(tf, ARG2);
 | |
| }
 | |
| 
 | |
| static Int expand_file_name3(USES_REGS1) {
 | |
|   Term tf = do_expand_file_name(Deref(ARG1), Deref(ARG2) PASS_REGS);
 | |
|   return Yap_unify(tf, ARG3);
 | |
| }
 | |
| 
 | |
| /*
 | |
|        static char *canoniseFileName( char *path) {
 | |
|        #if HAVE_REALPATH && HAVE_BASENAME
 | |
|        #if _WIN32 || defined(__MINGW32__)
 | |
|          char *o = malloc(YAP_FILENAME_MAX+1);
 | |
|          if (!o)
 | |
|            return NULL;
 | |
|          // first pass, remove Unix style stuff
 | |
|          if (unix2win(path, o, YAP_FILENAME_MAX) == NULL)
 | |
|            return NULL;
 | |
|          path = o;
 | |
|        #endif
 | |
|          char *rc;
 | |
|          if (tmp == NULL) return NULL;
 | |
|          rc = myrealpath(path);
 | |
|        #if _WIN32 || defined(__MINGW32__)
 | |
|          freeBuffer(o);
 | |
| #endif
 | |
|   return rc;
 | |
| #endif
 | |
| }
 | |
| */
 | |
| 
 | |
| static Int absolute_file_system_path(USES_REGS1) {
 | |
|   Term t = Deref(ARG1);
 | |
|   int l = push_text_stack();
 | |
|   const char *text = Yap_TextTermToText(t);
 | |
|   const char *fp;
 | |
|   bool rc;
 | |
| 
 | |
|   if (text == NULL) {
 | |
|     pop_text_stack(l);
 | |
|     return false;
 | |
|   }
 | |
|   if (!(fp = Yap_AbsoluteFile(text, true))) {
 | |
|     pop_text_stack(l);
 | |
|     return false;
 | |
|   }
 | |
|   rc = Yap_unify(Yap_MkTextTerm(fp, Yap_TextType(t)), ARG2);
 | |
|   pop_text_stack(l);
 | |
|   return rc;
 | |
| }
 | |
| 
 | |
| static Int prolog_to_os_filename(USES_REGS1) {
 | |
|   Term t = Deref(ARG1), t2 = Deref(ARG2);
 | |
|   char *fp;
 | |
|   char out[MAXPATHLEN + 1];
 | |
| 
 | |
|   if (IsVarTerm(t)) {
 | |
| 
 | |
|     if (IsVarTerm(t2)) {
 | |
|       Yap_Error(INSTANTIATION_ERROR, t, "prolog_to_os_filename");
 | |
|       return false;
 | |
|     } else if (IsAtomTerm(t2)) {
 | |
|       if (!(fp = PrologPath(RepAtom(AtomOfTerm(t2))->StrOfAE, out)))
 | |
|         return false;
 | |
|       return Yap_unify(ARG1, MkAtomTerm(Yap_LookupAtom(fp)));
 | |
|     } else {
 | |
|       Yap_Error(TYPE_ERROR_ATOM, t2, "prolog_to_os_filename");
 | |
|       return false;
 | |
|     }
 | |
|   } else if (!IsAtomTerm(t)) {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, t, "prolog_to_os_filename");
 | |
|     return false;
 | |
|   }
 | |
| 
 | |
|   if (!(fp = OsPath(RepAtom(AtomOfTerm(t))->StrOfAE, out)))
 | |
|     return false;
 | |
|   return Yap_unify(MkAtomTerm(Yap_LookupAtom(fp)), ARG2);
 | |
| }
 | |
| 
 | |
| Atom Yap_TemporaryFile(const char *prefix, int *fd) {
 | |
| #if HAVE_MKSTEMP
 | |
|   char *tmp = malloc(PATH_MAX);
 | |
|   int n;
 | |
|   int f;
 | |
|   if (tmp == NULL)
 | |
|     return NIL;
 | |
|   strncpy(tmp, prefix, PATH_MAX - 1);
 | |
|   n = strlen(tmp);
 | |
|   if (n >= 6 && tmp[n - 1] == 'X' && tmp[n - 2] == 'X' && tmp[n - 3] == 'X' &&
 | |
|       tmp[n - 4] == 'X' && tmp[n - 5] == 'X' && tmp[n - 6] == 'X')
 | |
|     f = mkstemp(tmp);
 | |
|   else {
 | |
|     strncat(tmp, "XXXXXX", PATH_MAX - 1);
 | |
|     f = mkstemp(tmp);
 | |
|   }
 | |
|   if (fd)
 | |
|     *fd = f;
 | |
|   return Yap_LookupAtom(tmp);
 | |
| #else
 | |
|   return AtomNil;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /** @pred make_directory(+ _Dir_)
 | |
| 
 | |
| Create a directory  _Dir_. The name of the directory must be an atom.
 | |
| 
 | |
| */
 | |
| static Int make_directory(USES_REGS1) {
 | |
|   const char *fd = AtomName(AtomOfTerm(ARG1));
 | |
| #if defined(__MINGW32__) || _MSC_VER
 | |
|   if (_mkdir(fd) == -1) {
 | |
| #else
 | |
|   if (mkdir(Yap_VFAlloc(fd), 0777) == -1) {
 | |
| #endif
 | |
|     /* return an error number */
 | |
|     return false; // errno?
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| static Int p_rmdir(USES_REGS1) {
 | |
|   const char *fd = Yap_VFAlloc(AtomName(AtomOfTerm(ARG1)));
 | |
| #if defined(__MINGW32__) || _MSC_VER
 | |
|   if (_rmdir(fd) == -1) {
 | |
| #else
 | |
|   if (rmdir(fd) == -1) {
 | |
| #endif
 | |
|     /* return an error number */
 | |
|     return (Yap_unify(ARG2, MkIntTerm(errno)));
 | |
|   }
 | |
|   return true;
 | |
| }
 | |
| 
 | |
| static bool initSysPath(Term tlib, Term tcommons, bool dir_done,
 | |
|                         bool commons_done) {
 | |
|   CACHE_REGS
 | |
| 
 | |
|   if (!Yap_unify(tlib, MkAtomTerm(Yap_LookupAtom(Yap_PLDIR))))
 | |
|     return false;
 | |
| 
 | |
|   return Yap_unify(tcommons, MkAtomTerm(Yap_LookupAtom(Yap_COMMONSDIR)));
 | |
| }
 | |
| 
 | |
| static Int libraries_directories(USES_REGS1) {
 | |
|   return initSysPath(ARG1, ARG2, false, false);
 | |
| }
 | |
| 
 | |
| static Int system_library(USES_REGS1) {
 | |
| #if __ANDROID__
 | |
|   static Term dir = 0;
 | |
|   Term t;
 | |
|   if (IsVarTerm(t = Deref(ARG1))) {
 | |
|     if (dir == 0)
 | |
|       return false;
 | |
|     return Yap_unify(dir, ARG1);
 | |
|   }
 | |
|   if (!IsAtomTerm(t))
 | |
|     return false;
 | |
|   dir = t;
 | |
|   return true;
 | |
| #else
 | |
|   return initSysPath(ARG1, MkVarTerm(), false, true);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| static Int commons_library(USES_REGS1) {
 | |
|   return initSysPath(MkVarTerm(), ARG1, true, false);
 | |
| }
 | |
| 
 | |
| static Int p_dir_sp(USES_REGS1) {
 | |
| #if ATARI || _MSC_VER || defined(__MINGW32__)
 | |
|   Term t = MkIntTerm('\\');
 | |
|   Term t2 = MkIntTerm('/');
 | |
| #else
 | |
|   Term t = MkIntTerm('/');
 | |
|   Term t2 = MkIntTerm('/');
 | |
| #endif
 | |
| 
 | |
|   return Yap_unify_constant(ARG1, t) || Yap_unify_constant(ARG1, t2);
 | |
| }
 | |
| 
 | |
| size_t Yap_InitPageSize(void) {
 | |
| #ifdef _WIN32
 | |
|   SYSTEM_INFO si;
 | |
|   GetSystemInfo(&si);
 | |
|   return si.dwPageSize;
 | |
| #elif HAVE_UNISTD_H
 | |
| #if defined(__FreeBSD__) || defined(__DragonFly__)
 | |
|   return getpagesize();
 | |
| #elif defined(_AIX)
 | |
|   return sysconf(_SC_PAGE_SIZE);
 | |
| #elif !defined(_SC_PAGESIZE)
 | |
|   return getpagesize();
 | |
| #else
 | |
|   return sysconf(_SC_PAGESIZE);
 | |
| #endif
 | |
| #else
 | |
|   bla bla
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /* TrueFileName -> Finds the true name of a file */
 | |
| 
 | |
| #ifdef __MINGW32__
 | |
| #include <ctype.h>
 | |
| #endif
 | |
| 
 | |
| static int volume_header(char *file) {
 | |
| #if _MSC_VER || defined(__MINGW32__)
 | |
|   char *ch = file;
 | |
|   int c;
 | |
| 
 | |
|   while ((c = ch[0]) != '\0') {
 | |
|     if (isalnum(c))
 | |
|       ch++;
 | |
|     else
 | |
|       return (c == ':');
 | |
|   }
 | |
| #endif
 | |
|   return (FALSE);
 | |
| }
 | |
| 
 | |
| int Yap_volume_header(char *file) { return volume_header(file); }
 | |
| 
 | |
| const char *Yap_getcwd(char *cwd, size_t cwdlen) {
 | |
|   if (GLOBAL_cwd && GLOBAL_cwd[0]) {
 | |
|     strcpy(cwd, GLOBAL_cwd);
 | |
|     return cwd;
 | |
|   }
 | |
| #if _WIN32 || defined(__MINGW32__)
 | |
|   if (GetCurrentDirectory(cwdlen, (char *)cwd) == 0) {
 | |
|     Yap_WinError("GetCurrentDirectory failed");
 | |
|     return NULL;
 | |
|   }
 | |
|   return (char *)cwd;
 | |
| #endif
 | |
|   const char *rc = getcwd(cwd, FILENAME_MAX);
 | |
|  // __android_log_print(ANDROID_LOG_INFO, "YAPDroid", "chdir %s", rc);
 | |
| return rc;
 | |
| }
 | |
| 
 | |
| static Int working_directory(USES_REGS1) {
 | |
|   char dir[YAP_FILENAME_MAX + 1];
 | |
|   Term t1 = Deref(ARG1), t2;
 | |
| 
 | |
|   if (!IsVarTerm(t1) && !IsAtomTerm(t1)) {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, t1, "working_directory");
 | |
|   }
 | |
|   if (!Yap_unify(t1,
 | |
|                  MkAtomTerm(Yap_LookupAtom(Yap_getcwd(dir, YAP_FILENAME_MAX)))))
 | |
|     return false;
 | |
|   t2 = Deref(ARG2);
 | |
|   if (IsVarTerm(t2)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, t2, "working_directory");
 | |
|   }
 | |
|   if (!IsAtomTerm(t2)) {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, t2, "working_directory");
 | |
|   }
 | |
|   if (t2 == TermEmptyAtom || t2 == TermDot)
 | |
|     return true;
 | |
|   return Yap_ChDir(RepAtom(AtomOfTerm(t2))->StrOfAE);
 | |
| }
 | |
| 
 | |
| static Int true_file_name(USES_REGS1) {
 | |
|   Term t = Deref(ARG1);
 | |
|   const char *s;
 | |
| 
 | |
|   if (IsVarTerm(t)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, t, "argument to true_file_name unbound");
 | |
|     return FALSE;
 | |
|   }
 | |
|   if (IsAtomTerm(t)) {
 | |
|     s = RepAtom(AtomOfTerm(t))->StrOfAE;
 | |
|   } else if (IsStringTerm(t)) {
 | |
|     s = StringOfTerm(t);
 | |
|   } else {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, t, "argument to true_file_name");
 | |
|     return FALSE;
 | |
|   }
 | |
|   int l = push_text_stack();
 | |
|   if (!(s = Yap_AbsoluteFile(s, true)))
 | |
|     return false;
 | |
|   bool rc = Yap_unify(ARG2, MkAtomTerm(Yap_LookupAtom(s)));
 | |
|   pop_text_stack(l);
 | |
|   return rc;
 | |
| }
 | |
| 
 | |
| static Int p_expand_file_name(USES_REGS1) {
 | |
|   Term t = Deref(ARG1);
 | |
|   const char *text, *text2;
 | |
| 
 | |
|   if (IsVarTerm(t)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, t, "argument to true_file_name unbound");
 | |
|     return FALSE;
 | |
|   }
 | |
|   int l = push_text_stack();
 | |
|   text = Yap_TextTermToText(t);
 | |
|   if (!text) {
 | |
|     pop_text_stack(l);
 | |
|     return false;
 | |
|   }
 | |
|   if (!(text2 = PlExpandVars(text, NULL))) {
 | |
|     pop_text_stack(l);
 | |
|     return false;
 | |
|   }
 | |
|   bool rc = Yap_unify(ARG2, Yap_MkTextTerm(text2, Yap_TextType(t)));
 | |
|   pop_text_stack(l);
 | |
|   return rc;
 | |
| }
 | |
| 
 | |
| static Int true_file_name3(USES_REGS1) {
 | |
|   Term t = Deref(ARG1), t2 = Deref(ARG2);
 | |
| 
 | |
|   if (IsVarTerm(t)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, t, "argument to true_file_name unbound");
 | |
|     return FALSE;
 | |
|   }
 | |
|   if (!IsAtomTerm(t)) {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, t, "argument to true_file_name");
 | |
|     return FALSE;
 | |
|   }
 | |
|   if (!IsVarTerm(t2)) {
 | |
|     if (!IsAtomTerm(t)) {
 | |
|       Yap_Error(TYPE_ERROR_ATOM, t2, "argument to true_file_name");
 | |
|       return FALSE;
 | |
|     }
 | |
|     //    root = RepAtom(AtomOfTerm(t2))->StrOfAE;
 | |
|   }
 | |
|   int lvl = push_text_stack();
 | |
|   const char *tmp = Yap_AbsoluteFile(RepAtom(AtomOfTerm(t))->StrOfAE, true);
 | |
|   Atom at = NULL;
 | |
|   bool rc = (tmp != NULL && (at = Yap_LookupAtom(tmp)) != NULL);
 | |
|   pop_text_stack(lvl);
 | |
|   return rc && Yap_unify(ARG3, MkAtomTerm(at));
 | |
| }
 | |
| 
 | |
| /* Executes $SHELL under Prolog */
 | |
| /** @pred  sh
 | |
| 
 | |
| 
 | |
|       Creates a new shell interaction.
 | |
| 
 | |
| 
 | |
|   */
 | |
| static Int p_sh(USES_REGS1) { /* sh				 */
 | |
| #ifdef HAVE_SYSTEM
 | |
|   char *shell;
 | |
|   shell = (char *)getenv("SHELL");
 | |
|   if (shell == NULL)
 | |
|     shell = "/bin/sh";
 | |
|   if (system(Yap_VFAlloc(shell)) < 0) {
 | |
| #if HAVE_STRERROR
 | |
|     Yap_Error(SYSTEM_ERROR_OPERATING_SYSTEM, TermNil, "%s in sh/0",
 | |
|               strerror(errno));
 | |
| #else
 | |
|     Yap_Error(SYSTEM_ERROR_OPERATING_SYSTEM, TermNil, "in sh/0");
 | |
| #endif
 | |
|     return FALSE;
 | |
|   }
 | |
|   return TRUE;
 | |
| #else
 | |
| #ifdef MSH
 | |
|   register char *shell;
 | |
|   shell = "msh -i";
 | |
|   system(shell);
 | |
|   return (TRUE);
 | |
| #else
 | |
|   Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
 | |
|             "sh not available in this configuration");
 | |
|   return (FALSE);
 | |
| #endif /* MSH */
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /** shell(+Command:text, -Status:integer) is det.
 | |
| 
 | |
|       Run an external command and wait for its completion.
 | |
|   */
 | |
| static Int p_shell(USES_REGS1) { /* '$shell'(+SystCommand) */
 | |
|   const char *cmd;
 | |
|   Term t1 = Deref(ARG1);
 | |
|   if (IsAtomTerm(t1))
 | |
|     cmd = RepAtom(AtomOfTerm(t1))->StrOfAE;
 | |
|   else if (IsStringTerm(t1))
 | |
|     cmd = StringOfTerm(t1);
 | |
|   else
 | |
|     return FALSE;
 | |
| #if _MSC_VER || defined(__MINGW32__)
 | |
|   {
 | |
|     int rval = system(cmd);
 | |
| 
 | |
|     return rval == 0;
 | |
|   }
 | |
| 
 | |
|   return true;
 | |
| #else
 | |
| #if HAVE_SYSTEM
 | |
|   char *shell;
 | |
|   register int bourne = FALSE;
 | |
| 
 | |
|   shell = (char *)getenv("SHELL");
 | |
|   if (!strcmp(shell, "/bin/sh"))
 | |
|     bourne = TRUE;
 | |
|   if (shell == NIL)
 | |
|     bourne = TRUE;
 | |
|   /* Yap_CloseStreams(TRUE); */
 | |
|   if (bourne)
 | |
|     return system(cmd) == 0;
 | |
|   else {
 | |
|     int status = -1;
 | |
|     int child = fork();
 | |
| 
 | |
|     if (child == 0) { /* let the children go */
 | |
|       if (!execl(shell, shell, "-c", cmd, NULL)) {
 | |
|         exit(-1);
 | |
|       }
 | |
|       exit(TRUE);
 | |
|     }
 | |
|     { /* put the father on wait */
 | |
|       int result = child < 0 ||
 | |
|                    /* vsc:I am not sure this is used, Stevens say wait returns
 | |
|                 an integer.
 | |
|                 #if NO_UNION_WAIT
 | |
|              */
 | |
|                    wait((&status)) != child ||
 | |
|                    /*
 | |
|                #else
 | |
|                wait ((union wait *) (&status)) != child ||
 | |
|                #endif
 | |
|              */
 | |
|                    status == 0;
 | |
|       return result;
 | |
|     }
 | |
|   }
 | |
| #else /* HAVE_SYSTEM */
 | |
| #ifdef MSH
 | |
|   register char *shell;
 | |
|   shell = "msh -i";
 | |
|   /* Yap_CloseStreams(); */
 | |
|   system(shell);
 | |
|   return TRUE;
 | |
| #else
 | |
|   Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
 | |
|             "shell not available in this configuration");
 | |
|   return FALSE;
 | |
| #endif
 | |
| #endif /* HAVE_SYSTEM */
 | |
| #endif /* _MSC_VER */
 | |
| }
 | |
| 
 | |
| /** system(+Command:text).
 | |
| 
 | |
|       Run an external command.
 | |
|   */
 | |
| 
 | |
| static Int p_system(USES_REGS1) { /* '$system'(+SystCommand)	       */
 | |
|   const char *cmd;
 | |
|   Term t1 = Deref(ARG1);
 | |
| 
 | |
|   if (IsVarTerm(t1)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, t1, "argument to system/1 unbound");
 | |
|     return FALSE;
 | |
|   } else if (IsAtomTerm(t1)) {
 | |
|     cmd = RepAtom(AtomOfTerm(t1))->StrOfAE;
 | |
|   } else if (IsStringTerm(t1)) {
 | |
|     cmd = StringOfTerm(t1);
 | |
|   } else {
 | |
|     if (!Yap_GetName(LOCAL_FileNameBuf, YAP_FILENAME_MAX, t1)) {
 | |
|       Yap_Error(TYPE_ERROR_ATOM, t1, "argument to system/1");
 | |
|       return false;
 | |
|     }
 | |
|     cmd = LOCAL_FileNameBuf;
 | |
|   }
 | |
| /* Yap_CloseStreams(TRUE); */
 | |
| #if _MSC_VER || defined(__MINGW32__)
 | |
| 
 | |
|   {
 | |
|     STARTUPINFO si;
 | |
|     PROCESS_INFORMATION pi;
 | |
| 
 | |
|     ZeroMemory(&si, sizeof(si));
 | |
|     si.cb = sizeof(si);
 | |
|     ZeroMemory(&pi, sizeof(pi));
 | |
| 
 | |
|     // Start the child process.
 | |
|     if (!CreateProcess(NULL,       // No module name (use command line)
 | |
|                        (LPSTR)cmd, // Command line
 | |
|                        NULL,       // Process handle not inheritable
 | |
|                        NULL,       // Thread handle not inheritable
 | |
|                        FALSE,      // Set handle inheritance to FALSE
 | |
|                        0,          // No creation flags
 | |
|                        NULL,       // Use parent's environment block
 | |
|                        NULL,       // Use parent's starting directory
 | |
|                        &si,        // Pointer to STARTUPINFO structure
 | |
|                        &pi)        // Pointer to PROCESS_INFORMATION structure
 | |
|     ) {
 | |
|       Yap_Error(SYSTEM_ERROR_INTERNAL, ARG1, "CreateProcess failed (%d).\n",
 | |
|                 GetLastError());
 | |
|       return FALSE;
 | |
|     }
 | |
|     // Wait until child process exits.
 | |
|     WaitForSingleObject(pi.hProcess, INFINITE);
 | |
| 
 | |
|     // Close process and thread handles.
 | |
|     CloseHandle(pi.hProcess);
 | |
|     CloseHandle(pi.hThread);
 | |
| 
 | |
|     return TRUE;
 | |
|   }
 | |
| 
 | |
|   return FALSE;
 | |
| #elif HAVE_SYSTEM
 | |
| #if _MSC_VER
 | |
|   _flushall();
 | |
| #endif
 | |
|   if (system(cmd)) {
 | |
| #if HAVE_STRERROR
 | |
|     Yap_Error(SYSTEM_ERROR_OPERATING_SYSTEM, t1, "%s in system(%s)",
 | |
|               strerror(errno), cmd);
 | |
| #else
 | |
|     Yap_Error(SYSTEM_ERROR_OPERATING_SYSTEM, t1, "in system(%s)", cmd);
 | |
| #endif
 | |
|     return FALSE;
 | |
|   }
 | |
|   return TRUE;
 | |
| #else
 | |
| #ifdef MSH
 | |
|   register char *shell;
 | |
|   shell = "msh -i";
 | |
|   /* Yap_CloseStreams(); */
 | |
|   system(shell);
 | |
|   return (TRUE);
 | |
| #undef command
 | |
| #else
 | |
|   Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil, "sh not available in this machine");
 | |
|   return (FALSE);
 | |
| #endif
 | |
| #endif /* HAVE_SYSTEM */
 | |
| }
 | |
| 
 | |
| static Int p_mv(USES_REGS1) { /* rename(+OldName,+NewName)   */
 | |
| #if HAVE_LINK
 | |
|   int r;
 | |
|   char *oldname, *newname;
 | |
|   Term t1 = Deref(ARG1);
 | |
|   Term t2 = Deref(ARG2);
 | |
|   if (IsVarTerm(t1)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, t1, "first argument to rename/2 unbound");
 | |
|   } else if (!IsAtomTerm(t1)) {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, t1, "first argument to rename/2 not atom");
 | |
|   }
 | |
|   if (IsVarTerm(t2)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, t2, "second argument to rename/2 unbound");
 | |
|   } else if (!IsAtomTerm(t2)) {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, t2, "second argument to rename/2 not atom");
 | |
|   } else {
 | |
|     oldname = RepAtom(AtomOfTerm(t1))->StrOfAE;
 | |
|     newname = RepAtom(AtomOfTerm(t2))->StrOfAE;
 | |
|     if ((r = link(oldname, newname)) == 0 && (r = unlink(oldname)) != 0)
 | |
|       unlink(newname);
 | |
|     if (r != 0) {
 | |
| #if HAVE_STRERROR
 | |
|       Yap_Error(SYSTEM_ERROR_OPERATING_SYSTEM, t2, "%s in rename(%s,%s)",
 | |
|                 strerror(errno), oldname, newname);
 | |
| #else
 | |
|       Yap_Error(SYSTEM_ERROR_OPERATING_SYSTEM, t2, "in rename(%s,%s)", oldname,
 | |
|                 newname);
 | |
| #endif
 | |
|       return false;
 | |
|     }
 | |
|     return true;
 | |
|   }
 | |
| #else
 | |
|   Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
 | |
|             "rename/2 not available in this machine");
 | |
| #endif
 | |
|   return false;
 | |
| }
 | |
| 
 | |
| #ifdef MAC
 | |
| 
 | |
| void Yap_SetTextFile(name) char *name;
 | |
| {
 | |
| #ifdef MACC
 | |
|   SetFileType(name, 'TEXT');
 | |
|   SetFileSignature(name, 'EDIT');
 | |
| #else
 | |
|   FInfo f;
 | |
|   FInfo *p = &f;
 | |
|   GetFInfo(name, 0, p);
 | |
|   p->fdType = 'TEXT';
 | |
| #ifdef MPW
 | |
|   if (mpwshell)
 | |
|     p->fdCreator = 'MPS\0';
 | |
| #endif
 | |
| #ifndef LIGHT
 | |
|   else
 | |
|     p->fdCreator = 'EDIT';
 | |
| #endif
 | |
|   SetFInfo(name, 0, p);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| #endif
 | |
| 
 | |
| /* return YAP's environment */
 | |
| static Int p_getenv(USES_REGS1) {
 | |
| #if HAVE_GETENV
 | |
|   Term t1 = Deref(ARG1), to;
 | |
|   char *s, *so;
 | |
| 
 | |
|   if (IsVarTerm(t1)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, t1, "first arg of getenv/2");
 | |
|     return (FALSE);
 | |
|   } else if (!IsAtomTerm(t1)) {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, t1, "first arg of getenv/2");
 | |
|     return (FALSE);
 | |
|   } else
 | |
|     s = RepAtom(AtomOfTerm(t1))->StrOfAE;
 | |
|   if ((so = getenv(s)) == NULL)
 | |
|     return (FALSE);
 | |
|   to = MkAtomTerm(Yap_LookupAtom(so));
 | |
|   return (Yap_unify_constant(ARG2, to));
 | |
| #else
 | |
|   Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
 | |
|             "getenv not available in this configuration");
 | |
|   return (FALSE);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /* set a variable in YAP's environment */
 | |
| static Int p_putenv(USES_REGS1) {
 | |
| #if HAVE_PUTENV
 | |
|   Term t1 = Deref(ARG1), t2 = Deref(ARG2);
 | |
|   char *s = "", *s2 = "", *p0, *p;
 | |
| 
 | |
|   if (IsVarTerm(t1)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, t1, "first arg to putenv/2");
 | |
|     return (FALSE);
 | |
|   } else if (!IsAtomTerm(t1)) {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, t1, "first arg to putenv/2");
 | |
|     return (FALSE);
 | |
|   } else
 | |
|     s = RepAtom(AtomOfTerm(t1))->StrOfAE;
 | |
|   if (IsVarTerm(t2)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, t1, "second arg to putenv/2");
 | |
|     return (FALSE);
 | |
|   } else if (!IsAtomTerm(t2)) {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, t2, "second arg to putenv/2");
 | |
|     return (FALSE);
 | |
|   } else
 | |
|     s2 = RepAtom(AtomOfTerm(t2))->StrOfAE;
 | |
|   while (!(p0 = p = Yap_AllocAtomSpace(strlen(s) + strlen(s2) + 3))) {
 | |
|     if (!Yap_growheap(FALSE, MinHeapGap, NULL)) {
 | |
|       Yap_Error(RESOURCE_ERROR_HEAP, TermNil, LOCAL_ErrorMessage);
 | |
|       return FALSE;
 | |
|     }
 | |
|   }
 | |
|   while ((*p++ = *s++) != '\0')
 | |
|     ;
 | |
|   p[-1] = '=';
 | |
|   while ((*p++ = *s2++) != '\0')
 | |
|     ;
 | |
|   if (putenv(p0) == 0)
 | |
|     return TRUE;
 | |
| #if HAVE_STRERROR
 | |
|   Yap_Error(SYSTEM_ERROR_OPERATING_SYSTEM, TermNil, "in putenv(%s)",
 | |
|             strerror(errno), p0);
 | |
| #else
 | |
|   Yap_Error(SYSTEM_ERROR_OPERATING_SYSTEM, TermNil, "in putenv(%s)", p0);
 | |
| #endif
 | |
|   return FALSE;
 | |
| #else
 | |
|   Yap_Error(SYSTEM_ERROR_INTERNAL, TermNil,
 | |
|             "putenv not available in this configuration");
 | |
|   return FALSE;
 | |
| #endif
 | |
| }
 | |
| 
 | |
| static Int p_host_type(USES_REGS1) {
 | |
|   Term out = MkAtomTerm(Yap_LookupAtom(HOST_ALIAS));
 | |
|   return (Yap_unify(out, ARG1));
 | |
| }
 | |
| 
 | |
| static Int p_yap_home(USES_REGS1) {
 | |
|   Term out;
 | |
| 
 | |
|   out = MkAtomTerm(Yap_LookupAtom(Yap_ROOTDIR));
 | |
|   return Yap_unify(out, ARG1);
 | |
| }
 | |
| 
 | |
| static Int p_yap_paths(USES_REGS1) {
 | |
|   Term out1, out2, out3;
 | |
| 
 | |
|   out1 = MkAtomTerm(Yap_LookupAtom(Yap_LIBDIR));
 | |
|   out2 = MkAtomTerm(Yap_LookupAtom(Yap_SHAREDIR));
 | |
|   out3 = MkAtomTerm(Yap_LookupAtom(Yap_BINDIR));
 | |
| 
 | |
|   return (Yap_unify(out1, ARG1) && Yap_unify(out2, ARG2) &&
 | |
|           Yap_unify(out3, ARG3));
 | |
| }
 | |
| 
 | |
| static Int p_log_event(USES_REGS1) {
 | |
|   Term in = Deref(ARG1);
 | |
|   Atom at;
 | |
| 
 | |
|   if (IsVarTerm(in))
 | |
|     return FALSE;
 | |
|   if (!IsAtomTerm(in))
 | |
|     return FALSE;
 | |
|   at = AtomOfTerm(in);
 | |
| #if DEBUG
 | |
|   if (IsBlob(at))
 | |
|     return FALSE;
 | |
|   else
 | |
|     fprintf(stderr, "LOG %s\n", RepAtom(at)->StrOfAE);
 | |
| #endif
 | |
|   if (IsBlob(at))
 | |
|     return false;
 | |
|   LOG(" %s ", RepAtom(at)->StrOfAE);
 | |
|   return TRUE;
 | |
| }
 | |
| 
 | |
| static Int p_env_separator(USES_REGS1) {
 | |
| #if defined(_WIN32)
 | |
|   return Yap_unify(MkIntegerTerm(';'), ARG1);
 | |
| #else
 | |
|   return Yap_unify(MkIntegerTerm(':'), ARG1);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * This is responsable for the initialization of all machine dependant
 | |
|  * predicates
 | |
|  */
 | |
| void Yap_InitSysbits(int wid) {
 | |
|   CACHE_REGS
 | |
| #if __simplescalar__
 | |
|   {
 | |
|     char *pwd = getenv("PWD");
 | |
|     strncpy(GLOBAL_pwd, pwd, YAP_FILENAME_MAX);
 | |
|   }
 | |
| #endif
 | |
|   Yap_InitWTime();
 | |
|   Yap_InitRandom();
 | |
|   /* let the caller control signals as it sees fit */
 | |
|   Yap_InitOSSignals(worker_id);
 | |
| }
 | |
| 
 | |
| static Int p_unix(USES_REGS1) {
 | |
| #ifdef unix
 | |
|   return TRUE;
 | |
| #else
 | |
| #ifdef __unix__
 | |
|   return TRUE;
 | |
| #else
 | |
| #ifdef __APPLE__
 | |
|   return TRUE;
 | |
| #else
 | |
|   return FALSE;
 | |
| #endif
 | |
| #endif
 | |
| #endif
 | |
| }
 | |
| 
 | |
| static Int p_win32(USES_REGS1) {
 | |
| #ifdef _WIN32
 | |
|   return TRUE;
 | |
| #else
 | |
| #ifdef __CYGWIN__
 | |
|   return TRUE;
 | |
| #else
 | |
|   return FALSE;
 | |
| #endif
 | |
| #endif
 | |
| }
 | |
| 
 | |
| static Int p_ld_path(USES_REGS1) {
 | |
|   return Yap_unify(ARG1, MkAtomTerm(Yap_LookupAtom(Yap_DLLDIR)));
 | |
| }
 | |
| 
 | |
| static Int p_address_bits(USES_REGS1) {
 | |
| #if SIZEOF_INT_P == 4
 | |
|   return Yap_unify(ARG1, MkIntTerm(32));
 | |
| #else
 | |
|   return Yap_unify(ARG1, MkIntTerm(64));
 | |
| #endif
 | |
| }
 | |
| 
 | |
| #ifdef _WIN32
 | |
| 
 | |
| /* This code is from SWI-Prolog by Jan Wielemaker */
 | |
| 
 | |
| #define wstreq(s, q) (wcscmp((s), (q)) == 0)
 | |
| 
 | |
| static HKEY reg_open_key(const wchar_t *which, int create) {
 | |
|   HKEY key = HKEY_CURRENT_USER;
 | |
|   DWORD disp;
 | |
|   LONG rval;
 | |
| 
 | |
|   while (*which) {
 | |
|     wchar_t buf[256];
 | |
|     wchar_t *s;
 | |
|     HKEY tmp;
 | |
| 
 | |
|     for (s = buf; *which && !(*which == '/' || *which == '\\');)
 | |
|       *s++ = *which++;
 | |
|     *s = '\0';
 | |
|     if (*which)
 | |
|       which++;
 | |
| 
 | |
|     if (wstreq(buf, L"HKEY_CLASSES_ROOT")) {
 | |
|       key = HKEY_CLASSES_ROOT;
 | |
|       continue;
 | |
|     } else if (wstreq(buf, L"HKEY_CURRENT_USER")) {
 | |
|       key = HKEY_CURRENT_USER;
 | |
|       continue;
 | |
|     } else if (wstreq(buf, L"HKEY_LOCAL_MACHINE")) {
 | |
|       key = HKEY_LOCAL_MACHINE;
 | |
|       continue;
 | |
|     } else if (wstreq(buf, L"HKEY_USERS")) {
 | |
|       key = HKEY_USERS;
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     if (RegOpenKeyExW(key, buf, 0L, KEY_READ, &tmp) == ERROR_SUCCESS) {
 | |
|       RegCloseKey(key);
 | |
|       key = tmp;
 | |
|       continue;
 | |
|     }
 | |
| 
 | |
|     if (!create)
 | |
|       return NULL;
 | |
| 
 | |
|     rval =
 | |
|         RegCreateKeyExW(key, buf, 0, L"", 0, KEY_ALL_ACCESS, NULL, &tmp, &disp);
 | |
|     RegCloseKey(key);
 | |
|     if (rval == ERROR_SUCCESS)
 | |
|       key = tmp;
 | |
|     else
 | |
|       return NULL;
 | |
|   }
 | |
| 
 | |
|   return key;
 | |
| }
 | |
| 
 | |
| #define MAXREGSTRLEN 1024
 | |
| 
 | |
| static wchar_t *WideStringFromAtom(Atom KeyAt USES_REGS) {
 | |
|   return Yap_AtomToWide(KeyAt);
 | |
| }
 | |
| 
 | |
| static Int p_win_registry_get_value(USES_REGS1) {
 | |
|   DWORD type;
 | |
|   BYTE data[MAXREGSTRLEN];
 | |
|   DWORD len = sizeof(data);
 | |
|   wchar_t *k, *name;
 | |
|   HKEY key;
 | |
|   Term Key = Deref(ARG1);
 | |
|   Term Name = Deref(ARG2);
 | |
|   Atom KeyAt, NameAt;
 | |
|   int l = push_text_stack();
 | |
| 
 | |
|   if (IsVarTerm(Key)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, Key,
 | |
|               "argument to win_registry_get_value unbound");
 | |
|     pop_text_stack(l);
 | |
|     return FALSE;
 | |
|   }
 | |
|   if (!IsAtomTerm(Key)) {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, Key, "argument to win_registry_get_value");
 | |
|     pop_text_stack(l);
 | |
|     return FALSE;
 | |
|   }
 | |
|   KeyAt = AtomOfTerm(Key);
 | |
|   if (IsVarTerm(Name)) {
 | |
|     Yap_Error(INSTANTIATION_ERROR, Key,
 | |
|               "argument to win_registry_get_value unbound");
 | |
|     pop_text_stack(l);
 | |
|     return FALSE;
 | |
|   }
 | |
|   if (!IsAtomTerm(Name)) {
 | |
|     Yap_Error(TYPE_ERROR_ATOM, Key, "argument to win_registry_get_value");
 | |
|     pop_text_stack(l);
 | |
|     return FALSE;
 | |
|   }
 | |
|   NameAt = AtomOfTerm(Name);
 | |
| 
 | |
|   k = WideStringFromAtom(KeyAt PASS_REGS);
 | |
|   if (!(key = reg_open_key(k, FALSE))) {
 | |
|     Yap_Error(EXISTENCE_ERROR_KEY, Key, "argument to win_registry_get_value");
 | |
|     pop_text_stack(l);
 | |
|     return FALSE;
 | |
|   }
 | |
|   name = WideStringFromAtom(NameAt PASS_REGS);
 | |
| 
 | |
|   if (RegQueryValueExW(key, name, NULL, &type, data, &len) == ERROR_SUCCESS) {
 | |
|     RegCloseKey(key);
 | |
|     switch (type) {
 | |
|     case REG_SZ:
 | |
|       ((wchar_t *)data)[len] = '\0';
 | |
|       Atom at = Yap_NWCharsToAtom((wchar_t *)data, len PASS_REGS);
 | |
|       pop_text_stack(l);
 | |
|       return Yap_unify(MkAtomTerm(at), ARG3);
 | |
|     case REG_DWORD: {
 | |
|       DWORD *d = (DWORD *)data;
 | |
|       pop_text_stack(l);
 | |
|       return Yap_unify(MkIntegerTerm((Int)d[0]), ARG3);
 | |
|     }
 | |
|     default:
 | |
|       pop_text_stack(l);
 | |
|       return FALSE;
 | |
|     }
 | |
|   }
 | |
|   pop_text_stack(l);
 | |
|   return FALSE;
 | |
| }
 | |
| 
 | |
| char *Yap_RegistryGetString(char *name) {
 | |
|   DWORD type;
 | |
|   BYTE data[MAXREGSTRLEN];
 | |
|   DWORD len = sizeof(data);
 | |
|   HKEY key;
 | |
|   char *ptr;
 | |
|   int i;
 | |
| 
 | |
| #if SIZEOF_INT_P == 8
 | |
|   if (!(key =
 | |
|             reg_open_key(L"HKEY_LOCAL_MACHINE/SOFTWARE/YAP/Prolog64", FALSE))) {
 | |
|     return NULL;
 | |
|   }
 | |
| #else
 | |
|   if (!(key = reg_open_key(L"HKEY_LOCAL_MACHINE/SOFTWARE/YAP/Prolog", FALSE))) {
 | |
|     return NULL;
 | |
|   }
 | |
| #endif
 | |
|   if (RegQueryValueEx(key, name, NULL, &type, data, &len) == ERROR_SUCCESS) {
 | |
|     RegCloseKey(key);
 | |
|     switch (type) {
 | |
|     case REG_SZ:
 | |
|       ptr = malloc(len + 2);
 | |
|       if (!ptr)
 | |
|         return NULL;
 | |
|       for (i = 0; i <= len; i++)
 | |
|         ptr[i] = data[i];
 | |
|       ptr[len + 1] = '\0';
 | |
|       return ptr;
 | |
|     default:
 | |
|       return NULL;
 | |
|     }
 | |
|   }
 | |
|   return NULL;
 | |
| }
 | |
| 
 | |
| #endif
 | |
| 
 | |
| void Yap_InitSysPreds(void) {
 | |
|   Yap_InitCPred("log_event", 1, p_log_event, SafePredFlag | SyncPredFlag);
 | |
|   Yap_InitCPred("sh", 0, p_sh, SafePredFlag | SyncPredFlag);
 | |
|   Yap_InitCPred("$shell", 1, p_shell, SafePredFlag | SyncPredFlag);
 | |
|   Yap_InitCPred("system", 1, p_system,
 | |
|                 SafePredFlag | SyncPredFlag | UserCPredFlag);
 | |
|   Yap_InitCPred("$rename", 2, p_mv, SafePredFlag | SyncPredFlag);
 | |
|   Yap_InitCPred("$yap_home", 1, p_yap_home, SafePredFlag);
 | |
|   Yap_InitCPred("$yap_paths", 3, p_yap_paths, SafePredFlag);
 | |
|   Yap_InitCPred("$dir_separator", 1, p_dir_sp, SafePredFlag);
 | |
|   Yap_InitCPred("libraries_directories", 2, libraries_directories, 0);
 | |
|     Yap_InitCPred("system_library", 1, system_library, 0);
 | |
|     Yap_InitCPred("exists_directory", 1, exists_directory, 0);
 | |
|   Yap_InitCPred("commons_library", 1, commons_library, 0);
 | |
|   Yap_InitCPred("$getenv", 2, p_getenv, SafePredFlag);
 | |
|   Yap_InitCPred("$putenv", 2, p_putenv, SafePredFlag | SyncPredFlag);
 | |
|   Yap_InitCPred("$host_type", 1, p_host_type, SafePredFlag | SyncPredFlag);
 | |
|   Yap_InitCPred("$env_separator", 1, p_env_separator, SafePredFlag);
 | |
|   Yap_InitCPred("$unix", 0, p_unix, SafePredFlag);
 | |
|   Yap_InitCPred("$win32", 0, p_win32, SafePredFlag);
 | |
|   Yap_InitCPred("$ld_path", 1, p_ld_path, SafePredFlag);
 | |
|   Yap_InitCPred("$address_bits", 1, p_address_bits, SafePredFlag);
 | |
|   Yap_InitCPred("$expand_file_name", 2, p_expand_file_name, SyncPredFlag);
 | |
|   Yap_InitCPred("expand_file_name", 3, expand_file_name3, SyncPredFlag);
 | |
|   Yap_InitCPred("expand_file_name", 2, expand_file_name, SyncPredFlag);
 | |
|   Yap_InitCPred("working_directory", 2, working_directory, SyncPredFlag);
 | |
|   Yap_InitCPred("prolog_to_os_filename", 2, prolog_to_os_filename,
 | |
|                 SyncPredFlag);
 | |
|   Yap_InitCPred("absolute_file_system_path", 2, absolute_file_system_path, 0);
 | |
|   Yap_InitCPred("real_path", 2, real_path, 0);
 | |
|   Yap_InitCPred("true_file_name", 2, true_file_name, SyncPredFlag);
 | |
|   Yap_InitCPred("true_file_name", 3, true_file_name3, SyncPredFlag);
 | |
| #ifdef _WIN32
 | |
|   Yap_InitCPred("win_registry_get_value", 3, p_win_registry_get_value, 0);
 | |
| #endif
 | |
|   Yap_InitCPred("rmdir", 2, p_rmdir, SyncPredFlag);
 | |
|   Yap_InitCPred("make_directory", 1, make_directory, SyncPredFlag);
 | |
| }
 |