| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | /*************************************************************************
 | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |  *									 * | 
					
						
							|  |  |  |  *	 YAP Prolog 							 * | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  *	Yap Prolog was developed at NCCUP - Universidade do Porto	 * | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  * Copyright L.Damas, V.S.Costa and Universidade do Porto 1985-1997	 * | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  ************************************************************************** | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  * File:		iopreds.c * | 
					
						
							|  |  |  |  * Last rev:	5/2/88							 * | 
					
						
							|  |  |  |  * mods: * | 
					
						
							|  |  |  |  * comments:	Input/Output C implemented predicates			 * | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  *************************************************************************/ | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #ifdef SCCS
 | 
					
						
							|  |  |  | static char SccsId[] = "%W% %G%"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /*
 | 
					
						
							|  |  |  |  * This file includes the definition of a miscellania of standard predicates | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |  * for yap refering to: Files and GLOBAL_Streams, Simple Input/Output, | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |  * | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-01-31 10:37:41 +00:00
										 |  |  | #include "sysbits.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | #if _MSC_VER || defined(__MINGW32__)
 | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #define SYSTEM_STAT _stat
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #define SYSTEM_STAT stat
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  | const char *Yap_GetFileName(Term t USES_REGS) { | 
					
						
							|  |  |  |   char *buf = Malloc(YAP_FILENAME_MAX + 1); | 
					
						
							|  |  |  |   if (IsApplTerm(t) && FunctorOfTerm(t) == FunctorSlash) { | 
					
						
							|  |  |  |     snprintf(buf, YAP_FILENAME_MAX, "%s/%s", Yap_GetFileName(ArgOfTerm(1, t)), | 
					
						
							|  |  |  |              Yap_GetFileName(ArgOfTerm(1, t))); | 
					
						
							| 
									
										
										
										
											2016-08-29 17:35:01 -05:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |   return Yap_TextTermToText(t PASS_REGS); | 
					
						
							| 
									
										
										
										
											2016-08-29 17:35:01 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * @pred file_name_extension( ? BaseFile, ?Extension, ?FullNameO) | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * Relate a file name with an extension. The extension is the filename's suffix | 
					
						
							|  |  |  |  * and indicates the kind of the file. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The predicate can be used to: | 
					
						
							|  |  |  |  * - Given __FullName__, extract the extension as _Extension_, and the remainder | 
					
						
							|  |  |  |  * as _BaseFile_. - Given _BaseFile_ and _?Extension_ obtain a _FullNameO_. | 
					
						
							|  |  |  |  * ~~~~ | 
					
						
							|  |  |  |  * ~~~~ | 
					
						
							|  |  |  |  *   Notice that: | 
					
						
							|  |  |  |  *   + if no suffix is found, file_name_extension/3 generates the empty | 
					
						
							|  |  |  |  * suffu]kx, `''`. + the extension does not include the `,` separator; + the | 
					
						
							|  |  |  |  * suffix may be longer thsn 3 characters + case should not matter in Windows | 
					
						
							|  |  |  |  * and MacOS + paths may not correspond to valid file names. | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * @return G | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int file_name_extension(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |   Term t1; | 
					
						
							|  |  |  |   Term t2; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   Term t3 = Deref(ARG3); | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |   int l = push_text_stack(); | 
					
						
							|  |  |  |   if (!IsVarTerm(t3)) { | 
					
						
							|  |  |  |     // full path is given.
 | 
					
						
							|  |  |  |     const char *f = Yap_GetFileName(t3); | 
					
						
							|  |  |  |     const char *ext; | 
					
						
							|  |  |  |     char *base; | 
					
						
							|  |  |  |     bool rc = true; | 
					
						
							|  |  |  |     seq_type_t typ = Yap_TextType(t3); | 
					
						
							|  |  |  |     if (!f) { | 
					
						
							|  |  |  |       pop_text_stack(l); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-02 16:04:34 +01:00
										 |  |  |     size_t len_b = strlen(f), lenb_b; | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     char *candidate = strrchr(f, '.'); | 
					
						
							|  |  |  |     char *file = strrchr(f, '/'); | 
					
						
							|  |  |  |     if (candidate && file && candidate > file) { | 
					
						
							| 
									
										
										
										
											2017-10-02 16:04:34 +01:00
										 |  |  |       lenb_b = candidate - f; | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |       ext = candidate + 1; | 
					
						
							| 
									
										
										
										
											2016-08-30 08:26:14 -05:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |       lenb_b = len_b; | 
					
						
							|  |  |  |       ext = ""; | 
					
						
							| 
									
										
										
										
											2016-08-30 08:26:14 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     base = Malloc(lenb_b + 1); | 
					
						
							|  |  |  |     memcpy(base, f, lenb_b); | 
					
						
							|  |  |  |     base[lenb_b] = '\0'; | 
					
						
							|  |  |  |     if (IsVarTerm(t1 = Deref(ARG1))) { | 
					
						
							|  |  |  |       // should always succeed
 | 
					
						
							|  |  |  |       rc = Yap_unify(t1, Yap_MkTextTerm(base, typ)); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |       char *f_a = (char *)Yap_GetFileName(t1 PASS_REGS); | 
					
						
							| 
									
										
										
										
											2016-08-29 17:35:01 -05:00
										 |  |  | #if __APPLE__ || _WIN32
 | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |       rc = strcasecmp(f_a, base) == 0; | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2017-10-02 16:04:34 +01:00
										 |  |  |       rc = strcmp(f_a, base) == 0; | 
					
						
							| 
									
										
										
										
											2016-08-29 17:35:01 -05:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     if (rc) { | 
					
						
							|  |  |  |       if (IsVarTerm(t2 = Deref(ARG2))) { | 
					
						
							|  |  |  |         // should always succeed
 | 
					
						
							|  |  |  |         rc = Yap_unify(t2, Yap_MkTextTerm(ext, typ)); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  |         char *f_a = (char *)Yap_TextTermToText(t2 PASS_REGS); | 
					
						
							|  |  |  |         if (f_a[0] == '.') { | 
					
						
							|  |  |  |           f_a += 1; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2016-08-29 17:35:01 -05:00
										 |  |  | #if __APPLE__ || _WIN32
 | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |         rc = strcasecmp(f_a, ext) == 0; | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2017-10-02 16:04:34 +01:00
										 |  |  |         rc = strcmp(f_a, ext) == 0; | 
					
						
							| 
									
										
										
										
											2016-08-29 17:35:01 -05:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-08-29 17:35:01 -05:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     pop_text_stack(l); | 
					
						
							|  |  |  |     return rc; | 
					
						
							| 
									
										
										
										
											2016-08-29 17:35:01 -05:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     const char *f; | 
					
						
							| 
									
										
										
										
											2016-08-29 17:35:01 -05:00
										 |  |  |     char *f2; | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     seq_type_t typ, typ1 = Yap_TextType((t1 = Deref(ARG1))), | 
					
						
							|  |  |  |                     typ2 = Yap_TextType((t2 = Deref(ARG2))); | 
					
						
							|  |  |  |     if (typ1 == typ2) { | 
					
						
							|  |  |  |       typ = typ1; | 
					
						
							|  |  |  |     } else if (typ1 == YAP_STRING_ATOM || typ2 == YAP_STRING_ATOM) { | 
					
						
							|  |  |  |       typ = YAP_STRING_ATOM; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       typ = YAP_STRING_STRING; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!(f = Yap_TextTermToText(t1 PASS_REGS))) { | 
					
						
							|  |  |  |       pop_text_stack(l); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     if (!(f2 = (char *)Yap_TextTermToText(t2 PASS_REGS))) { | 
					
						
							|  |  |  |       pop_text_stack(l); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     if (f2[0] == '.') { | 
					
						
							|  |  |  |       f2++; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     size_t lenb_b = strlen(f); | 
					
						
							|  |  |  |     char *o = Realloc((void *)f, lenb_b + strlen(f2) + 2); | 
					
						
							|  |  |  |     o[lenb_b] = '.'; | 
					
						
							|  |  |  |     o += lenb_b + 1; | 
					
						
							|  |  |  |     pop_text_stack(l); | 
					
						
							|  |  |  |     return strcpy(o, f2) && (t3 = Yap_MkTextTerm(o, typ)) && | 
					
						
							|  |  |  |            Yap_unify(t3, ARG3); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int access_path(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   Term tname = Deref(ARG1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (IsVarTerm(tname)) { | 
					
						
							|  |  |  |     Yap_Error(INSTANTIATION_ERROR, tname, "access"); | 
					
						
							| 
									
										
										
										
											2016-04-12 16:04:33 +01:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   } else if (!IsAtomTerm(tname)) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     Yap_Error(TYPE_ERROR_ATOM, tname, "access"); | 
					
						
							| 
									
										
										
										
											2016-04-12 16:04:33 +01:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   } else { | 
					
						
							|  |  |  | #if HAVE_STAT
 | 
					
						
							|  |  |  |     struct SYSTEM_STAT ss; | 
					
						
							| 
									
										
										
										
											2016-04-12 16:04:33 +01:00
										 |  |  |     char *file_name; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     file_name = RepAtom(AtomOfTerm(tname))->StrOfAE; | 
					
						
							|  |  |  |     if (SYSTEM_STAT(file_name, &ss) != 0) { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |       /* ignore errors while checking a file */ | 
					
						
							| 
									
										
										
										
											2016-04-12 16:04:33 +01:00
										 |  |  |       return true; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-04-12 16:04:33 +01:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-04-12 16:04:33 +01:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int exists_file(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   Term tname = Deref(ARG1); | 
					
						
							|  |  |  |   char *file_name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (IsVarTerm(tname)) { | 
					
						
							|  |  |  |     Yap_Error(INSTANTIATION_ERROR, tname, "access"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   } else if (!IsAtomTerm(tname)) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     Yap_Error(TYPE_ERROR_ATOM, tname, "access"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  | #if HAVE_STAT
 | 
					
						
							|  |  |  |     struct SYSTEM_STAT ss; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     file_name = RepAtom(AtomOfTerm(tname))->StrOfAE; | 
					
						
							|  |  |  |     if (SYSTEM_STAT(file_name, &ss) != 0) { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |       /* ignore errors while checking a file */ | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | #if _MSC_VER
 | 
					
						
							|  |  |  |     return ss.st_mode & S_IFREG; | 
					
						
							| 
									
										
										
										
											2016-02-28 19:32:55 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     return S_ISREG(ss.st_mode); | 
					
						
							| 
									
										
										
										
											2016-02-28 19:32:55 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #else
 | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int file_exists(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2016-01-04 14:11:09 +00:00
										 |  |  |   Term tname = Deref(ARG1); | 
					
						
							|  |  |  |   char *file_name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (IsVarTerm(tname)) { | 
					
						
							|  |  |  |     Yap_Error(INSTANTIATION_ERROR, tname, "access"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   } else if (!IsAtomTerm(tname)) { | 
					
						
							| 
									
										
										
										
											2016-01-04 14:11:09 +00:00
										 |  |  |     Yap_Error(TYPE_ERROR_ATOM, tname, "access"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  | #if HAVE_STAT
 | 
					
						
							|  |  |  |     struct SYSTEM_STAT ss; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     file_name = RepAtom(AtomOfTerm(tname))->StrOfAE; | 
					
						
							|  |  |  |     if (SYSTEM_STAT(file_name, &ss) != 0) { | 
					
						
							|  |  |  |       if (errno == ENOENT) | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |         return false; | 
					
						
							|  |  |  |       PlIOError(SYSTEM_ERROR_OPERATING_SYSTEM, tname, "error %s", | 
					
						
							|  |  |  |                 strerror(errno)); | 
					
						
							| 
									
										
										
										
											2016-01-04 14:11:09 +00:00
										 |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int time_file(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   Term tname = Deref(ARG1); | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   if (IsVarTerm(tname)) { | 
					
						
							|  |  |  |     Yap_Error(INSTANTIATION_ERROR, tname, "access"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   } else if (!IsAtomTerm(tname)) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     Yap_Error(TYPE_ERROR_ATOM, tname, "access"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     const char *n = RepAtom(AtomOfTerm(tname))->StrOfAE; | 
					
						
							|  |  |  | #if __WIN32
 | 
					
						
							| 
									
										
										
										
											2016-01-31 10:37:41 +00:00
										 |  |  |     FILETIME ft; | 
					
						
							|  |  |  |     HANDLE hdl; | 
					
						
							|  |  |  |     Term rc; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-04-17 17:09:10 -07:00
										 |  |  |     if ((hdl = CreateFile(n, 0, 0, NULL, OPEN_EXISTING, 0, 0)) == 0) { | 
					
						
							|  |  |  |       Yap_WinError("in time_file"); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       return false; | 
					
						
							| 
									
										
										
										
											2016-04-17 17:09:10 -07:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (GetFileTime(hdl, NULL, NULL, &ft) == 0) { | 
					
						
							|  |  |  |       Yap_WinError("in time_file"); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       return false; | 
					
						
							| 
									
										
										
										
											2016-04-17 17:09:10 -07:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     // Convert the last-write time to local time.
 | 
					
						
							|  |  |  |     // FileTimeToSystemTime(&ftWrite, &stUTC);
 | 
					
						
							|  |  |  |     // SystemTimeToTzSpecificLocalTime(NULL, &stUTC, &stLocal);
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     CloseHandle(hdl); | 
					
						
							|  |  |  |     ULONGLONG qwResult; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Copy the time into a quadword.
 | 
					
						
							|  |  |  |     qwResult = (((ULONGLONG)ft.dwHighDateTime) << 32) + ft.dwLowDateTime; | 
					
						
							|  |  |  | #if SIZEOF_INT_P == 8
 | 
					
						
							|  |  |  |     rc = MkIntegerTerm(qwResult); | 
					
						
							|  |  |  | #elif USE_GMP
 | 
					
						
							|  |  |  |     char s[64]; | 
					
						
							|  |  |  |     MP_INT rop; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     snprintf(s, 64, "%I64d", (long long int)n); | 
					
						
							|  |  |  |     mpz_init_set_str(&rop, s, 10); | 
					
						
							| 
									
										
										
										
											2016-07-31 10:15:55 -05:00
										 |  |  |     rc = Yap_MkBigIntTerm((void *)&rop PASS_REGS); | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-01-31 10:37:41 +00:00
										 |  |  |     rc = MkIntegerTerm(ft.dwHighDateTime); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     return Yap_unify(ARG2, rc); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #elif HAVE_STAT
 | 
					
						
							|  |  |  |     struct SYSTEM_STAT ss; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     if (SYSTEM_STAT(n, &ss) != 0) { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |       /* ignore errors while checking a file */ | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return Yap_unify(ARG2, MkIntegerTerm(ss.st_mtime)); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int file_size(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2015-09-29 23:44:11 +01:00
										 |  |  |   int rc; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   Int sno = Yap_CheckStream( | 
					
						
							|  |  |  |       ARG1, (Input_Stream_f | Output_Stream_f | Socket_Stream_f), | 
					
						
							|  |  |  |       "file_size/2"); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   if (sno < 0) | 
					
						
							|  |  |  |     return (FALSE); | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   if (GLOBAL_Stream[sno].status & Seekable_Stream_f && | 
					
						
							|  |  |  |       !(GLOBAL_Stream[sno].status & | 
					
						
							|  |  |  |         (InMemory_Stream_f | Socket_Stream_f | Pipe_Stream_f))) { | 
					
						
							|  |  |  |     // there
 | 
					
						
							|  |  |  |     struct stat file_stat; | 
					
						
							|  |  |  |     if ((rc = fstat(fileno(GLOBAL_Stream[sno].file), &file_stat)) < 0) { | 
					
						
							|  |  |  |       UNLOCK(GLOBAL_Stream[sno].streamlock); | 
					
						
							| 
									
										
										
										
											2015-09-29 23:44:11 +01:00
										 |  |  |       if (rc == ENOENT) | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |         PlIOError(EXISTENCE_ERROR_SOURCE_SINK, ARG1, "%s in file_size", | 
					
						
							|  |  |  |                   strerror(errno)); | 
					
						
							| 
									
										
										
										
											2015-09-29 23:44:11 +01:00
										 |  |  |       else | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |         PlIOError(PERMISSION_ERROR_INPUT_STREAM, ARG1, "%s in file_size", | 
					
						
							|  |  |  |                   strerror(errno)); | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     // and back again
 | 
					
						
							| 
									
										
										
										
											2015-06-19 10:10:02 +01:00
										 |  |  |     UNLOCK(GLOBAL_Stream[sno].streamlock); | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     return Yap_unify_constant(ARG2, MkIntegerTerm(file_stat.st_size)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   UNLOCK(GLOBAL_Stream[sno].streamlock); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-21 14:47:05 -05:00
										 |  |  | static Int lines_in_file(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2016-09-27 12:32:48 -05:00
										 |  |  |   Int sno = Yap_CheckStream(ARG1, (Input_Stream_f), "lines_in_file/2"); | 
					
						
							| 
									
										
										
										
											2016-09-21 14:47:05 -05:00
										 |  |  |   if (sno < 0) | 
					
						
							| 
									
										
										
										
											2016-09-27 12:32:48 -05:00
										 |  |  |     return false; | 
					
						
							|  |  |  |   FILE *f = GLOBAL_Stream[sno].file; | 
					
						
							|  |  |  |   size_t count = 0; | 
					
						
							|  |  |  |   int ch; | 
					
						
							| 
									
										
										
										
											2016-09-21 14:47:05 -05:00
										 |  |  | #if __ANDROID__
 | 
					
						
							|  |  |  | #define getw getc
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2016-09-27 12:32:48 -05:00
										 |  |  |   if (!f) | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   while ((ch = getw(f)) >= 0) { | 
					
						
							|  |  |  |     if (ch == '\n') { | 
					
						
							|  |  |  |       count++; | 
					
						
							| 
									
										
										
										
											2016-09-21 14:47:05 -05:00
										 |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-09-27 12:32:48 -05:00
										 |  |  |   return Yap_unify(ARG2, MkIntegerTerm(count)); | 
					
						
							| 
									
										
										
										
											2016-09-21 14:47:05 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int access_file(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   Term tname = Deref(ARG1); | 
					
						
							|  |  |  |   Term tmode = Deref(ARG2); | 
					
						
							| 
									
										
										
										
											2016-02-11 06:15:30 -08:00
										 |  |  |   char *ares; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   Atom atmode; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (IsVarTerm(tmode)) { | 
					
						
							|  |  |  |     Yap_Error(INSTANTIATION_ERROR, tmode, "access_file/2"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   } else if (!IsAtomTerm(tmode)) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     Yap_Error(TYPE_ERROR_ATOM, tname, "access_file/2"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   atmode = AtomOfTerm(tmode); | 
					
						
							|  |  |  |   if (IsVarTerm(tname)) { | 
					
						
							|  |  |  |     Yap_Error(INSTANTIATION_ERROR, tname, "access_file/2"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   } else if (!IsAtomTerm(tname)) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     Yap_Error(TYPE_ERROR_ATOM, tname, "access_file/2"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     if (atmode == AtomNone) | 
					
						
							|  |  |  |       return TRUE; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     if (!(ares = RepAtom(AtomOfTerm(tname))->StrOfAE)) | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #if HAVE_ACCESS
 | 
					
						
							| 
									
										
										
										
											2016-02-28 19:32:55 +00:00
										 |  |  | #if _WIN32
 | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     int mode; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-31 10:15:55 -05:00
										 |  |  |     if (atmode == AtomExist) | 
					
						
							|  |  |  |       mode = 00; | 
					
						
							|  |  |  |     else if (atmode == AtomExists) | 
					
						
							|  |  |  |       mode = 00; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     else if (atmode == AtomWrite) | 
					
						
							|  |  |  |       mode = 02; | 
					
						
							|  |  |  |     else if (atmode == AtomRead) | 
					
						
							|  |  |  |       mode = 04; | 
					
						
							|  |  |  |     else if (atmode == AtomAppend) | 
					
						
							|  |  |  |       mode = 03; | 
					
						
							|  |  |  |     else if (atmode == AtomCsult) | 
					
						
							|  |  |  |       mode = 04; | 
					
						
							|  |  |  |     else if (atmode == AtomExecute) | 
					
						
							|  |  |  |       mode = 00; // can always execute?
 | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |       Yap_Error(DOMAIN_ERROR_IO_MODE, tmode, "access_file/2"); | 
					
						
							|  |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (access(ares, mode) < 0) { | 
					
						
							|  |  |  |       /* ignore errors while checking a file */ | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-02-28 19:32:55 +00:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   { | 
					
						
							|  |  |  |     int mode; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-31 10:15:55 -05:00
										 |  |  |     if (atmode == AtomExist) | 
					
						
							|  |  |  |       mode = F_OK; | 
					
						
							|  |  |  |     else if (atmode == AtomExists) | 
					
						
							|  |  |  |       mode = F_OK; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     else if (atmode == AtomWrite) | 
					
						
							|  |  |  |       mode = W_OK; | 
					
						
							|  |  |  |     else if (atmode == AtomRead) | 
					
						
							|  |  |  |       mode = R_OK; | 
					
						
							|  |  |  |     else if (atmode == AtomAppend) | 
					
						
							|  |  |  |       mode = W_OK; | 
					
						
							|  |  |  |     else if (atmode == AtomCsult) | 
					
						
							|  |  |  |       mode = R_OK; | 
					
						
							|  |  |  |     else if (atmode == AtomExecute) | 
					
						
							|  |  |  |       mode = X_OK; | 
					
						
							|  |  |  |     else { | 
					
						
							| 
									
										
										
										
											2016-02-11 06:15:30 -08:00
										 |  |  |       Yap_Error(DOMAIN_ERROR_IO_MODE, tmode, "access_file/2"); | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |       return FALSE; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     if (access(ares, mode) < 0) { | 
					
						
							|  |  |  |       /* ignore errors while checking a file */ | 
					
						
							| 
									
										
										
										
											2016-02-11 06:15:30 -08:00
										 |  |  |       return false; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-02-11 06:15:30 -08:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-02-28 19:32:55 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #elif HAVE_STAT
 | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     struct SYSTEM_STAT ss; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (SYSTEM_STAT(ares, &ss) != 0) { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |       /* ignore errors while checking a file */ | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return TRUE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   return FALSE; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int exists_directory(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   Term tname = Deref(ARG1); | 
					
						
							|  |  |  |   char *file_name; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (IsVarTerm(tname)) { | 
					
						
							|  |  |  |     Yap_Error(INSTANTIATION_ERROR, tname, "exists_directory/1"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   } else if (!IsAtomTerm(tname)) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     Yap_Error(TYPE_ERROR_ATOM, tname, "exists_directory/1"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  | #if HAVE_STAT
 | 
					
						
							|  |  |  |     struct SYSTEM_STAT ss; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     file_name = RepAtom(AtomOfTerm(tname))->StrOfAE; | 
					
						
							|  |  |  |     if (SYSTEM_STAT(file_name, &ss) != 0) { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |       /* ignore errors while checking a file */ | 
					
						
							| 
									
										
										
										
											2016-02-11 06:15:30 -08:00
										 |  |  |       return false; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     return (S_ISDIR(ss.st_mode)); | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int is_absolute_file_name(USES_REGS1) { /* file_base_name(Stream,N) */ | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   Term t = Deref(ARG1); | 
					
						
							|  |  |  |   Atom at; | 
					
						
							| 
									
										
										
										
											2016-05-30 11:24:40 +01:00
										 |  |  |   bool rc; | 
					
						
							| 
									
										
										
										
											2016-11-02 00:16:36 -05:00
										 |  |  |   if (IsVarTerm(t)) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     Yap_Error(INSTANTIATION_ERROR, t, "file_base_name/2"); | 
					
						
							| 
									
										
										
										
											2016-05-30 11:24:40 +01:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-09-26 15:15:15 +01:00
										 |  |  |   int l = push_text_stack(); | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |   const char *buf = Yap_TextTermToText(t PASS_REGS); | 
					
						
							| 
									
										
										
										
											2016-02-13 03:11:25 +00:00
										 |  |  |   if (buf) { | 
					
						
							| 
									
										
										
										
											2016-05-30 11:24:40 +01:00
										 |  |  |     rc = Yap_IsAbsolutePath(buf); | 
					
						
							| 
									
										
										
										
											2016-02-13 03:11:25 +00:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     at = AtomOfTerm(t); | 
					
						
							| 
									
										
										
										
											2016-05-30 11:24:40 +01:00
										 |  |  | #if _WIN32
 | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     rc = PathIsRelative(RepAtom(at)->StrOfAE); | 
					
						
							| 
									
										
										
										
											2016-05-30 11:24:40 +01:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     rc = RepAtom(at)->StrOfAE[0] == '/'; | 
					
						
							| 
									
										
										
										
											2016-07-31 10:15:55 -05:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-09-26 15:15:15 +01:00
										 |  |  |   pop_text_stack(l); | 
					
						
							| 
									
										
										
										
											2016-07-31 10:15:55 -05:00
										 |  |  |   return rc; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int file_base_name(USES_REGS1) { /* file_base_name(Stream,N) */ | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   Term t = Deref(ARG1); | 
					
						
							|  |  |  |   Atom at; | 
					
						
							|  |  |  |   if (IsVarTerm(t)) { | 
					
						
							|  |  |  |     Yap_Error(INSTANTIATION_ERROR, t, "file_base_name/2"); | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     return FALSE; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   } | 
					
						
							|  |  |  |   at = AtomOfTerm(t); | 
					
						
							| 
									
										
										
										
											2016-11-02 00:16:36 -05:00
										 |  |  |   const char *c = RepAtom(at)->StrOfAE; | 
					
						
							|  |  |  |   const char *s; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | #if HAVE_BASENAME && 0 // DISABLED: Linux basename is not compatible with
 | 
					
						
							|  |  |  |                        // file_base_name in SWI and GNU
 | 
					
						
							| 
									
										
										
										
											2016-11-02 00:16:36 -05:00
										 |  |  |   char c1[YAP_FILENAME_MAX + 1]; | 
					
						
							|  |  |  |   strncpy(c1, c, YAP_FILENAME_MAX); | 
					
						
							|  |  |  |   s = basename(c1); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-11-02 00:16:36 -05:00
										 |  |  |   Int i = strlen(c); | 
					
						
							|  |  |  |   while (i && !Yap_dir_separator((int)c[--i])) | 
					
						
							|  |  |  |     ; | 
					
						
							|  |  |  |   if (Yap_dir_separator((int)c[i])) { | 
					
						
							|  |  |  |     i++; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-11-02 00:16:36 -05:00
										 |  |  |   s = c + i; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   return Yap_unify(ARG2, MkAtomTerm(Yap_LookupAtom(s))); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int file_directory_name(USES_REGS1) { /* file_directory_name(Stream,N) */ | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   Term t = Deref(ARG1); | 
					
						
							|  |  |  |   Atom at; | 
					
						
							|  |  |  |   if (IsVarTerm(t)) { | 
					
						
							|  |  |  |     Yap_Error(INSTANTIATION_ERROR, t, "file_directory_name/2"); | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   } | 
					
						
							|  |  |  |   at = AtomOfTerm(t); | 
					
						
							| 
									
										
										
										
											2016-11-02 00:16:36 -05:00
										 |  |  |   const char *c = RepAtom(at)->StrOfAE; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | #if HAVE_BASENAME && 0 // DISABLED: Linux basename is not compatible with
 | 
					
						
							|  |  |  |                        // file_base_name in SWI and GNU
 | 
					
						
							| 
									
										
										
										
											2016-11-02 00:16:36 -05:00
										 |  |  |   const char *s; | 
					
						
							|  |  |  |   char c1[YAP_FILENAME_MAX + 1]; | 
					
						
							|  |  |  |   strncpy(c1, c, YAP_FILENAME_MAX); | 
					
						
							|  |  |  |   s = dirname(c1); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-11-02 00:16:36 -05:00
										 |  |  |   char s[YAP_FILENAME_MAX + 1]; | 
					
						
							|  |  |  |   Int i = strlen(c); | 
					
						
							|  |  |  |   strncpy(s, c, YAP_FILENAME_MAX); | 
					
						
							|  |  |  |   while (--i) { | 
					
						
							|  |  |  |     if (Yap_dir_separator((int)c[i])) | 
					
						
							|  |  |  |       break; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |   if (i == 0) { | 
					
						
							|  |  |  |     s[0] = '.'; | 
					
						
							|  |  |  |     i = 1; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2016-11-02 00:16:36 -05:00
										 |  |  |   s[i] = '\0'; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |   return Yap_unify(ARG2, MkAtomTerm(Yap_LookupAtom(s))); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | static Int same_file(USES_REGS1) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   char *f1 = RepAtom(AtomOfTerm(Deref(ARG1)))->StrOfAE; | 
					
						
							|  |  |  |   char *f2 = RepAtom(AtomOfTerm(Deref(ARG2)))->StrOfAE; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   if (strcmp(f1, f2) == 0) | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     return TRUE; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | #if HAVE_LSTAT
 | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |   { | 
					
						
							|  |  |  |     int out; | 
					
						
							|  |  |  |     struct stat *b1, *b2; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     while ((char *)HR + sizeof(struct stat) * 2 > (char *)(ASP - 1024)) { | 
					
						
							|  |  |  |       if (!Yap_gcl(2 * sizeof(struct stat), 2, ENV, Yap_gcP())) { | 
					
						
							|  |  |  |         Yap_Error(RESOURCE_ERROR_STACK, TermNil, LOCAL_ErrorMessage); | 
					
						
							|  |  |  |         return FALSE; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     b1 = (struct stat *)HR; | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     b2 = b1 + 1; | 
					
						
							|  |  |  |     if (strcmp(f1, "user_input") == 0) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       if (fstat(fileno(GLOBAL_Stream[0].file), b1) == -1) { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |         /* file does not exist, but was opened? Return -1 */ | 
					
						
							|  |  |  |         return FALSE; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     } else if (strcmp(f1, "user_output") == 0) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       if (fstat(fileno(GLOBAL_Stream[1].file), b1) == -1) { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |         /* file does not exist, but was opened? Return -1 */ | 
					
						
							|  |  |  |         return FALSE; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     } else if (strcmp(f1, "user_error") == 0) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       if (fstat(fileno(GLOBAL_Stream[2].file), b1) == -1) { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |         /* file does not exist, but was opened? Return -1 */ | 
					
						
							|  |  |  |         return FALSE; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |     } else if (stat(f1, b1) == -1) { | 
					
						
							|  |  |  |       /* file does not exist, but was opened? Return -1 */ | 
					
						
							|  |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     if (strcmp(f2, "user_input") == 0) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       if (fstat(fileno(GLOBAL_Stream[0].file), b2) == -1) { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |         /* file does not exist, but was opened? Return -1 */ | 
					
						
							|  |  |  |         return FALSE; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     } else if (strcmp(f2, "user_output") == 0) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       if (fstat(fileno(GLOBAL_Stream[1].file), b2) == -1) { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |         /* file does not exist, but was opened? Return -1 */ | 
					
						
							|  |  |  |         return FALSE; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |     } else if (strcmp(f2, "user_error") == 0) { | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       if (fstat(fileno(GLOBAL_Stream[2].file), b2) == -1) { | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |         /* file does not exist, but was opened? Return -1 */ | 
					
						
							|  |  |  |         return FALSE; | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |       } | 
					
						
							|  |  |  |     } else if (stat(f2, b2) == -1) { | 
					
						
							|  |  |  |       /* file does not exist, but was opened? Return -1 */ | 
					
						
							|  |  |  |       return FALSE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     out = (b1->st_ino == b2->st_ino | 
					
						
							|  |  |  | #ifdef __LCC__
 | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |            && memcmp((const void *)&(b1->st_dev), (const void *)&(b2->st_dev), | 
					
						
							|  |  |  |                      sizeof(buf1.st_dev)) == 0 | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |            && b1->st_dev == b2->st_dev | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2017-10-02 08:58:51 +01:00
										 |  |  |     ); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  |     return out; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   return (FALSE); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  | void Yap_InitFiles(void) { | 
					
						
							|  |  |  |   Yap_InitCPred("file_base_name", 2, file_base_name, SafePredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("file_directory_name", 2, file_directory_name, SafePredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("is_absolute_file_name", 1, is_absolute_file_name, | 
					
						
							|  |  |  |                 SafePredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("same_file", 2, same_file, SafePredFlag | SyncPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("$access_file", 2, access_file, SafePredFlag | SyncPredFlag); | 
					
						
							| 
									
										
										
										
											2016-09-21 14:47:05 -05:00
										 |  |  |   Yap_InitCPred("$lines_in_file", 2, lines_in_file, | 
					
						
							|  |  |  |                 SafePredFlag | SyncPredFlag); | 
					
						
							| 
									
										
										
										
											2016-02-29 03:13:23 +00:00
										 |  |  |   Yap_InitCPred("access", 1, access_path, SafePredFlag | SyncPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("exists_directory", 1, exists_directory, | 
					
						
							|  |  |  |                 SafePredFlag | SyncPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("exists_file", 1, exists_file, SafePredFlag | SyncPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("$file_exists", 1, file_exists, SafePredFlag | SyncPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("time_file64", 2, time_file, SafePredFlag | SyncPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("time_file", 2, time_file, SafePredFlag | SyncPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("file_size", 2, file_size, SafePredFlag | SyncPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("file_name_extension", 3, file_name_extension, | 
					
						
							|  |  |  |                 SafePredFlag | SyncPredFlag); | 
					
						
							| 
									
										
										
										
											2015-06-18 01:33:55 +01:00
										 |  |  | } |