| 
									
										
										
										
											2001-04-09 19:54:03 +00: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:		load_foreign.c					 * | 
					
						
							|  |  |  |  * comments:	dynamic loader of external routines			 * | 
					
						
							|  |  |  |  *									 * | 
					
						
							|  |  |  |  *************************************************************************/ | 
					
						
							|  |  |  | #ifdef SCCS
 | 
					
						
							|  |  |  | static char     SccsId[] = "%W% %G%.2"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Yap.h"
 | 
					
						
							|  |  |  | #include "Yatom.h"
 | 
					
						
							| 
									
										
										
										
											2009-10-23 14:22:17 +01:00
										 |  |  | #include "YapHeap.h"
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #include "yapio.h"
 | 
					
						
							|  |  |  | #include <stdlib.h>
 | 
					
						
							|  |  |  | #if HAVE_STRING_H
 | 
					
						
							|  |  |  | #include <string.h>
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "Foreign.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-11 17:38:10 +00:00
										 |  |  | #if _WIN32 || defined(__CYGWIN__)
 | 
					
						
							| 
									
										
										
										
											2009-10-23 09:09:16 +01:00
										 |  |  | #ifndef YAP_SHLIB_SUFFIX
 | 
					
						
							|  |  |  | #define YAP_SHLIB_SUFFIX ".dll"
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | STD_PROTO(Int p_load_foreign, (void)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Int | 
					
						
							|  |  |  | p_load_foreign(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-01-23 17:57:56 +00:00
										 |  |  |   StringList ofiles = NULL; | 
					
						
							|  |  |  |   StringList libs = NULL; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   char *InitProcName; | 
					
						
							|  |  |  |   YapInitProc InitProc = NULL; | 
					
						
							|  |  |  |   Term t, t1; | 
					
						
							|  |  |  |   StringList new; | 
					
						
							|  |  |  |   Int returncode = FALSE; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   strcpy(Yap_ErrorSay,"Invalid arguments"); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /* collect the list of object files */ | 
					
						
							|  |  |  |   t = Deref(ARG1); | 
					
						
							|  |  |  |   while(1) { | 
					
						
							|  |  |  |     if (t == TermNil) break; | 
					
						
							|  |  |  |     t1 = HeadOfTerm(t); | 
					
						
							|  |  |  |     t = TailOfTerm(t); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     new = (StringList) Yap_AllocCodeSpace(sizeof(StringListItem)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     new->next = ofiles; | 
					
						
							| 
									
										
										
										
											2010-12-12 18:28:55 +00:00
										 |  |  |     new->name = AtomOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     ofiles = new; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* collect the list of library files */ | 
					
						
							|  |  |  |   t = Deref(ARG2); | 
					
						
							|  |  |  |   while(1) { | 
					
						
							|  |  |  |     if (t == TermNil) break; | 
					
						
							|  |  |  |     t1 = HeadOfTerm(t); | 
					
						
							|  |  |  |     t = TailOfTerm(t); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     new = (StringList) Yap_AllocCodeSpace(sizeof(StringListItem)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     new->next = libs; | 
					
						
							| 
									
										
										
										
											2010-12-12 18:28:55 +00:00
										 |  |  |     new->name = AtomOfTerm(t1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     libs = new; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* get the initialization function name */ | 
					
						
							|  |  |  |   t1 = Deref(ARG3); | 
					
						
							|  |  |  |   InitProcName = RepAtom(AtomOfTerm(t1))->StrOfAE; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   /* call the OS specific function for dynamic loading */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   if(Yap_LoadForeign(ofiles,libs,InitProcName,&InitProc)==LOAD_SUCCEEDED) { | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     (*InitProc)(); | 
					
						
							|  |  |  |     returncode = TRUE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   /* I should recover space if load foreign fails */ | 
					
						
							|  |  |  |   if (returncode == TRUE) { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     ForeignObj *f_code = (ForeignObj *)Yap_AllocCodeSpace(sizeof(ForeignObj)); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     f_code->objs = ofiles; | 
					
						
							|  |  |  |     f_code->libs = libs; | 
					
						
							|  |  |  |     f_code->f = InitProcName; | 
					
						
							|  |  |  |     f_code->next = ForeignCodeLoaded; | 
					
						
							|  |  |  |     f_code->module = CurrentModule; | 
					
						
							| 
									
										
										
										
											2010-04-12 17:21:50 +01:00
										 |  |  |     ForeignCodeLoaded = f_code; | 
					
						
							| 
									
										
										
										
											2008-01-23 17:57:56 +00:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     while (ofiles) { | 
					
						
							|  |  |  |       new = ofiles->next; | 
					
						
							|  |  |  |       Yap_FreeCodeSpace((ADDR)ofiles); | 
					
						
							|  |  |  |       ofiles = new; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     while (libs) { | 
					
						
							|  |  |  |       new = libs->next; | 
					
						
							|  |  |  |       Yap_FreeCodeSpace((ADDR)libs); | 
					
						
							|  |  |  |       libs = new; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |   } | 
					
						
							|  |  |  |   return returncode; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  | static Int | 
					
						
							|  |  |  | p_open_shared_object(void) { | 
					
						
							|  |  |  |   Term t = Deref(ARG1); | 
					
						
							|  |  |  |   Term tflags = Deref(ARG2); | 
					
						
							| 
									
										
										
										
											2010-11-29 18:44:39 +00:00
										 |  |  |   char *s; | 
					
						
							|  |  |  |   void *handle; | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |   if (IsVarTerm(t)) { | 
					
						
							|  |  |  |     Yap_Error(INSTANTIATION_ERROR,t,"open_shared_object/3"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   }  | 
					
						
							|  |  |  |   if (!IsAtomTerm(t)) { | 
					
						
							|  |  |  |     Yap_Error(TYPE_ERROR_ATOM,t,"open_shared_object/3"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  |   if (IsVarTerm(tflags)) { | 
					
						
							|  |  |  |     Yap_Error(INSTANTIATION_ERROR,tflags,"open_shared_object/3"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   }  | 
					
						
							| 
									
										
										
										
											2010-11-29 18:44:39 +00:00
										 |  |  |   if (!IsIntegerTerm(tflags)) { | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |     Yap_Error(TYPE_ERROR_INTEGER,tflags,"open_shared_object/3"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |    | 
					
						
							| 
									
										
										
										
											2010-11-29 18:44:39 +00:00
										 |  |  |   s = RepAtom(AtomOfTerm(t))->StrOfAE; | 
					
						
							|  |  |  |   if ((handle = Yap_LoadForeignFile(s, IntegerOfTerm(tflags)))==NULL) { | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |   } else { | 
					
						
							| 
									
										
										
										
											2010-11-29 18:44:39 +00:00
										 |  |  |     return Yap_unify(MkIntegerTerm((Int)handle),ARG3); | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int | 
					
						
							|  |  |  | p_close_shared_object(void) { | 
					
						
							|  |  |  |   Term t = Deref(ARG1); | 
					
						
							|  |  |  |   void *handle; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (IsVarTerm(t)) { | 
					
						
							| 
									
										
										
										
											2010-12-01 19:30:58 +00:00
										 |  |  |     Yap_Error(INSTANTIATION_ERROR,t,"close_shared_object/1"); | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |     return FALSE; | 
					
						
							|  |  |  |   }  | 
					
						
							|  |  |  |   if (!IsIntegerTerm(t)) { | 
					
						
							| 
									
										
										
										
											2010-12-01 19:30:58 +00:00
										 |  |  |     Yap_Error(TYPE_ERROR_INTEGER,t,"close_shared_object/1"); | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-11-29 18:44:39 +00:00
										 |  |  |   handle = (char *)IntegerOfTerm(t); | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |   | 
					
						
							| 
									
										
										
										
											2010-11-29 18:44:39 +00:00
										 |  |  |   return Yap_CloseForeignFile(handle); | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int | 
					
						
							|  |  |  | p_call_shared_object_function(void) { | 
					
						
							|  |  |  |   Term t = Deref(ARG1); | 
					
						
							|  |  |  |   Term tfunc = Deref(ARG2); | 
					
						
							| 
									
										
										
										
											2010-12-01 19:30:58 +00:00
										 |  |  |   Term tmod; | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |   void *handle; | 
					
						
							| 
									
										
										
										
											2010-12-01 19:30:58 +00:00
										 |  |  |   Term OldCurrentModule = CurrentModule; | 
					
						
							|  |  |  |   Int res; | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-12-01 19:30:58 +00:00
										 |  |  |   tmod = CurrentModule; | 
					
						
							|  |  |  |  restart: | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |   if (IsVarTerm(t)) { | 
					
						
							| 
									
										
										
										
											2010-12-01 19:30:58 +00:00
										 |  |  |     Yap_Error(INSTANTIATION_ERROR,t,"call_shared_object_function/2"); | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |     return FALSE; | 
					
						
							| 
									
										
										
										
											2010-12-01 19:30:58 +00:00
										 |  |  |   } else if (IsApplTerm(t)) { | 
					
						
							|  |  |  |     Functor    fun = FunctorOfTerm(t); | 
					
						
							|  |  |  |     if (fun == FunctorModule) { | 
					
						
							|  |  |  |       tmod = ArgOfTerm(1, t); | 
					
						
							|  |  |  |       if (IsVarTerm(tmod) ) { | 
					
						
							|  |  |  | 	Yap_Error(INSTANTIATION_ERROR,t,"call_shared_object_function/2"); | 
					
						
							|  |  |  | 	return FALSE; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (!IsAtomTerm(tmod) ) { | 
					
						
							|  |  |  | 	Yap_Error(TYPE_ERROR_ATOM,ARG1,"call_shared_object_function/2"); | 
					
						
							|  |  |  | 	return FALSE; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       t = ArgOfTerm(2, t); | 
					
						
							|  |  |  |       goto restart; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else if (!IsIntegerTerm(t)) { | 
					
						
							|  |  |  |     Yap_Error(TYPE_ERROR_INTEGER,t,"call_shared_object_function/2"); | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-11-29 18:44:39 +00:00
										 |  |  |   handle = (void *)IntegerOfTerm(t); | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |   if (IsVarTerm(tfunc)) { | 
					
						
							| 
									
										
										
										
											2010-12-01 19:30:58 +00:00
										 |  |  |     Yap_Error(INSTANTIATION_ERROR,t,"call_shared_object_function/2"); | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |     return FALSE; | 
					
						
							|  |  |  |   }  | 
					
						
							|  |  |  |   if (!IsAtomTerm(tfunc)) { | 
					
						
							| 
									
										
										
										
											2010-12-01 19:30:58 +00:00
										 |  |  |     Yap_Error(TYPE_ERROR_ATOM,t,"call_shared_object_function/2/3"); | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |     return FALSE; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-12-01 19:30:58 +00:00
										 |  |  |   CurrentModule = tmod; | 
					
						
							|  |  |  |   res = Yap_CallForeignFile(handle, RepAtom(AtomOfTerm(tfunc))->StrOfAE); | 
					
						
							|  |  |  |   CurrentModule = OldCurrentModule; | 
					
						
							|  |  |  |   return res; | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | static Int | 
					
						
							|  |  |  | p_obj_suffix(void) { | 
					
						
							| 
									
										
										
										
											2010-06-14 09:27:23 +01:00
										 |  |  |   return Yap_unify(Yap_StringToList(YAP_SHLIB_SUFFIX),ARG1); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-06-14 22:43:15 +01:00
										 |  |  | static Int | 
					
						
							|  |  |  | p_open_shared_objects(void) { | 
					
						
							|  |  |  | #ifdef YAP_SHLIB_SUFFIX
 | 
					
						
							|  |  |  |   return TRUE; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |   return FALSE; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_InitLoadForeign(void) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   if (Yap_argv == NULL) | 
					
						
							|  |  |  |     Yap_FindExecutable("yap"); | 
					
						
							| 
									
										
										
										
											2002-09-23 17:06:13 +00:00
										 |  |  |   else | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     Yap_FindExecutable(Yap_argv[0]); | 
					
						
							| 
									
										
										
										
											2004-11-18 22:32:40 +00:00
										 |  |  |   Yap_InitCPred("$load_foreign_files", 3, p_load_foreign, SafePredFlag|SyncPredFlag|HiddenPredFlag); | 
					
						
							| 
									
										
										
										
											2010-06-14 22:43:15 +01:00
										 |  |  |   Yap_InitCPred("$open_shared_objects", 0, p_open_shared_objects, SafePredFlag|HiddenPredFlag); | 
					
						
							| 
									
										
										
										
											2010-06-17 00:29:01 +01:00
										 |  |  |   Yap_InitCPred("$open_shared_object", 3, p_open_shared_object, SyncPredFlag|HiddenPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("close_shared_object", 1, p_close_shared_object, SyncPredFlag|SafePredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("call_shared_object_function", 2, p_call_shared_object_function, SyncPredFlag); | 
					
						
							| 
									
										
										
										
											2004-11-18 22:32:40 +00:00
										 |  |  |   Yap_InitCPred("$obj_suffix", 1, p_obj_suffix, SafePredFlag|HiddenPredFlag); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void  | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_ReOpenLoadForeign(void) | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   ForeignObj *f_code = ForeignCodeLoaded; | 
					
						
							| 
									
										
										
										
											2004-02-12 12:37:12 +00:00
										 |  |  |   Term OldModule = CurrentModule; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   while (f_code != NULL) { | 
					
						
							| 
									
										
										
										
											2010-12-07 18:21:10 +00:00
										 |  |  |     YapInitProc InitProc = NULL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2001-11-15 00:01:43 +00:00
										 |  |  |     CurrentModule = f_code->module; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |     if(Yap_ReLoadForeign(f_code->objs,f_code->libs,f_code->f,&InitProc)==LOAD_SUCCEEDED) { | 
					
						
							| 
									
										
										
										
											2010-11-22 18:07:01 +00:00
										 |  |  |       if (InitProc) | 
					
						
							|  |  |  | 	(*InitProc)(); | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  |     } | 
					
						
							|  |  |  |     f_code = f_code->next; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2001-11-15 00:01:43 +00:00
										 |  |  |   CurrentModule = OldModule; | 
					
						
							| 
									
										
										
										
											2001-04-09 19:54:03 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 |