| 
									
										
										
										
											2002-06-04 18:21:55 +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:		agc.c							 * | 
					
						
							|  |  |  | * Last rev:								 * | 
					
						
							|  |  |  | * mods:									 * | 
					
						
							|  |  |  | * comments:	reclaim unused atoms and functors			 * | 
					
						
							|  |  |  | *									 * | 
					
						
							|  |  |  | *************************************************************************/ | 
					
						
							|  |  |  | #ifdef SCCS
 | 
					
						
							|  |  |  | static char     SccsId[] = "@(#)agc.c	1.3 3/15/90"; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "absmi.h"
 | 
					
						
							|  |  |  | #include "alloc.h"
 | 
					
						
							| 
									
										
										
										
											2002-06-05 14:23:15 +00:00
										 |  |  | #include "yapio.h"
 | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  | #include "iopreds.h"
 | 
					
						
							| 
									
										
										
										
											2005-05-27 21:44:00 +00:00
										 |  |  | #include "attvar.h"
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | #ifdef DEBUG
 | 
					
						
							| 
									
										
										
										
											2005-05-30 05:26:50 +00:00
										 |  |  | /* #define DEBUG_RESTORE1 1 */ | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | /* #define DEBUG_RESTORE2 1 */ | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  | /* #define DEBUG_RESTORE3 1 */ | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | #define errout Yap_stderr
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | STATIC_PROTO(void  RestoreEntries, (PropEntry *)); | 
					
						
							| 
									
										
										
										
											2004-02-06 10:19:49 +00:00
										 |  |  | STATIC_PROTO(void  CleanCode, (PredEntry *)); | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  | static int agc_calls; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  | static YAP_ULONG_LONG agc_collected; | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static Int tot_agc_time = 0; /* total time spent in GC */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int tot_agc_recovered = 0; /* number of heap objects in all garbage collections */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | #define AtomMarkedBit 1
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline void | 
					
						
							|  |  |  | MarkAtomEntry(AtomEntry *ae) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   CELL c = (CELL)(ae->NextOfAE); | 
					
						
							|  |  |  |   c |= AtomMarkedBit; | 
					
						
							|  |  |  |   ae->NextOfAE = (Atom)c; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline int | 
					
						
							|  |  |  | AtomResetMark(AtomEntry *ae) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   CELL c = (CELL)(ae->NextOfAE); | 
					
						
							|  |  |  |   if (c & AtomMarkedBit) { | 
					
						
							|  |  |  |     c &= ~AtomMarkedBit; | 
					
						
							|  |  |  |     ae->NextOfAE = (Atom)c; | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |     return TRUE; | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |   return FALSE; | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline Atom | 
					
						
							|  |  |  | CleanAtomMarkedBit(Atom a) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   CELL c = (CELL)a; | 
					
						
							|  |  |  |   c &= ~AtomMarkedBit; | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |   return (Atom)c; | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline Functor | 
					
						
							|  |  |  | FuncAdjust(Functor f) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-05-30 03:26:37 +00:00
										 |  |  |   if (!IsExtensionFunctor(f)) {   | 
					
						
							|  |  |  |     AtomEntry *ae = RepAtom(NameOfFunctor(f)); | 
					
						
							|  |  |  |     MarkAtomEntry(ae); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  |   return(f); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline Term | 
					
						
							|  |  |  | AtomTermAdjust(Term t) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   AtomEntry *ae = RepAtom(AtomOfTerm(t)); | 
					
						
							|  |  |  |   MarkAtomEntry(ae); | 
					
						
							|  |  |  |   return(t);   | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static inline Atom | 
					
						
							|  |  |  | AtomAdjust(Atom a) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   AtomEntry *ae; | 
					
						
							|  |  |  |   if (a == NIL) return(a); | 
					
						
							|  |  |  |   ae = RepAtom(a); | 
					
						
							|  |  |  |   MarkAtomEntry(ae); | 
					
						
							|  |  |  |   return(a); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define IsOldCode(P) FALSE
 | 
					
						
							|  |  |  | #define IsOldCodeCellPtr(P) FALSE
 | 
					
						
							|  |  |  | #define IsOldDelay(P) FALSE
 | 
					
						
							|  |  |  | #define IsOldDelayPtr(P) FALSE
 | 
					
						
							|  |  |  | #define IsOldLocalInTR(P) FALSE
 | 
					
						
							|  |  |  | #define IsOldLocalInTRPtr(P) FALSE
 | 
					
						
							|  |  |  | #define IsOldGlobal(P) FALSE
 | 
					
						
							|  |  |  | #define IsOldGlobalPtr(P) FALSE
 | 
					
						
							|  |  |  | #define IsOldTrail(P) FALSE
 | 
					
						
							|  |  |  | #define IsOldTrailPtr(P) FALSE
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define CharP(X) ((char *)(X))
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define AddrAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2007-09-28 23:18:17 +00:00
										 |  |  | #define PredEntryAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | #define AtomEntryAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2007-01-08 08:27:19 +00:00
										 |  |  | #define GlobalEntryAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | #define BlobTermAdjust(P) (P)
 | 
					
						
							|  |  |  | #define CellPtoHeapAdjust(P) (P)
 | 
					
						
							|  |  |  | #define CellPtoHeapCellAdjust(P) (P)
 | 
					
						
							|  |  |  | #define CellPtoTRAdjust(P) (P)
 | 
					
						
							|  |  |  | #define CodeAddrAdjust(P) (P)
 | 
					
						
							|  |  |  | #define ConsultObjAdjust(P) (P)
 | 
					
						
							|  |  |  | #define DelayAddrAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2006-08-22 16:12:46 +00:00
										 |  |  | #define DelayAdjust(P) (P)
 | 
					
						
							|  |  |  | #define GlobalAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | #define DBRefAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2003-01-21 16:14:52 +00:00
										 |  |  | #define DBRefPAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2004-02-06 02:26:23 +00:00
										 |  |  | #define DBTermAdjust(P) (P)
 | 
					
						
							|  |  |  | #define LUIndexAdjust(P) (P)
 | 
					
						
							|  |  |  | #define SIndexAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | #define LocalAddrAdjust(P) (P)
 | 
					
						
							|  |  |  | #define GlobalAddrAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2004-02-06 02:26:23 +00:00
										 |  |  | #define PtoLUCAdjust(P) (P)
 | 
					
						
							|  |  |  | #define PtoStCAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | #define PtoArrayEAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2005-10-28 17:38:50 +00:00
										 |  |  | #define PtoArraySAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2006-08-22 16:12:46 +00:00
										 |  |  | #define PtoGlobalEAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | #define PtoDelayAdjust(P) (P)
 | 
					
						
							|  |  |  | #define PtoGloAdjust(P) (P)
 | 
					
						
							|  |  |  | #define PtoLocAdjust(P) (P)
 | 
					
						
							|  |  |  | #define PtoHeapCellAdjust(P) (P)
 | 
					
						
							|  |  |  | #define PtoOpAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2006-10-10 14:08:17 +00:00
										 |  |  | #define PtoLUClauseAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  | #define PtoLUIndexAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2007-11-06 17:02:13 +00:00
										 |  |  | #define PtoDBTLAdjust(P) (P)
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | #define PtoPredAdjust(P) (P)
 | 
					
						
							|  |  |  | #define PropAdjust(P) (P)
 | 
					
						
							|  |  |  | #define TrailAddrAdjust(P) (P)
 | 
					
						
							|  |  |  | #define XAdjust(P) (P)
 | 
					
						
							|  |  |  | #define YAdjust(P) (P)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | recompute_mask(DBRef dbr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   return; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void  | 
					
						
							|  |  |  | rehash(CELL *oldcode, int NOfE, int KindOfEntries) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "rheap.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  | static void init_reg_copies(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   OldASP = ASP; | 
					
						
							|  |  |  |   OldLCL0 = LCL0; | 
					
						
							|  |  |  |   OldTR = TR; | 
					
						
							|  |  |  |   OldGlobalBase = (CELL *)Yap_GlobalBase; | 
					
						
							|  |  |  |   OldH = H; | 
					
						
							|  |  |  |   OldH0 = H0; | 
					
						
							|  |  |  |   OldTrailBase = Yap_TrailBase; | 
					
						
							|  |  |  |   OldTrailTop = Yap_TrailTop; | 
					
						
							|  |  |  |   OldHeapBase = Yap_HeapBase; | 
					
						
							|  |  |  |   OldHeapTop = HeapTop; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | mark_hash_entry(AtomHashEntry *HashPtr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Atom atm; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   atm = HashPtr->Entry; | 
					
						
							|  |  |  |   if (atm) { | 
					
						
							|  |  |  |     AtomEntry      *at =  RepAtom(atm); | 
					
						
							|  |  |  |     do { | 
					
						
							|  |  |  | #ifdef DEBUG_RESTORE1			/* useful during debug */
 | 
					
						
							| 
									
										
										
										
											2006-12-13 16:10:26 +00:00
										 |  |  |       if (IsWideAtom(atm)) | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  | 	fprintf(errout, "Restoring %S\n", at->WStrOfAE); | 
					
						
							| 
									
										
										
										
											2006-12-13 16:10:26 +00:00
										 |  |  |       else | 
					
						
							|  |  |  | 	fprintf(errout, "Restoring %s\n", at->StrOfAE); | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  |       RestoreEntries(RepProp(at->PropsOfAE)); | 
					
						
							|  |  |  |       atm = at->NextOfAE; | 
					
						
							|  |  |  |       at = RepAtom(CleanAtomMarkedBit(atm)); | 
					
						
							|  |  |  |     } while (!EndOfPAEntr(at)); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-09-28 23:18:17 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | mark_hash_preds(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   UInt i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i = 0; i < PredHashTableSize; i++) { | 
					
						
							|  |  |  |     PredEntry *p = PredHash[i]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (p) { | 
					
						
							|  |  |  |       Prop nextp = p->NextOfPE = PropAdjust(p->NextOfPE); | 
					
						
							|  |  |  |       CleanCode(p); | 
					
						
							|  |  |  |       p = RepPredProp(nextp); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This is the really tough part, to restore the whole of the heap  | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void  | 
					
						
							|  |  |  | mark_atoms(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   AtomHashEntry *HashPtr = HashChain; | 
					
						
							|  |  |  |   register int    i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   restore_codes(); | 
					
						
							| 
									
										
										
										
											2003-10-28 01:16:03 +00:00
										 |  |  |   for (i = 0; i < AtomHashTableSize; ++i) { | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  |     mark_hash_entry(HashPtr); | 
					
						
							|  |  |  |     HashPtr++; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   HashPtr = WideHashChain; | 
					
						
							|  |  |  |   for (i = 0; i < WideAtomHashTableSize; ++i) { | 
					
						
							|  |  |  |     mark_hash_entry(HashPtr); | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  |     HashPtr++; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |   mark_hash_entry(&INVISIBLECHAIN); | 
					
						
							| 
									
										
										
										
											2007-09-28 23:18:17 +00:00
										 |  |  |   mark_hash_preds(); | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | mark_trail(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2007-03-18 23:09:12 +00:00
										 |  |  |   register tr_fr_ptr pt; | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-03-18 23:09:12 +00:00
										 |  |  |   pt = TR; | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  |   /* moving the trail is simple */ | 
					
						
							| 
									
										
										
										
											2007-03-18 23:09:12 +00:00
										 |  |  |   while (pt != (tr_fr_ptr)Yap_TrailBase) { | 
					
						
							|  |  |  |     CELL reg = TrailTerm(pt-1); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  |     if (!IsVarTerm(reg)) { | 
					
						
							|  |  |  |       if (IsAtomTerm(reg)) { | 
					
						
							|  |  |  | 	MarkAtomEntry(RepAtom(AtomOfTerm(reg))); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2007-03-18 23:09:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     pt--; | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | mark_local(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   register CELL   *pt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Adjusting the local */ | 
					
						
							|  |  |  |   pt = LCL0; | 
					
						
							|  |  |  |   /* moving the trail is simple */ | 
					
						
							|  |  |  |   while (pt > ASP) { | 
					
						
							|  |  |  |     CELL reg = *--pt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!IsVarTerm(reg)) { | 
					
						
							| 
									
										
										
										
											2007-03-18 23:09:12 +00:00
										 |  |  |       if (IsAtomTerm(reg) | 
					
						
							|  |  |  | #if TABLING
 | 
					
						
							|  |  |  | 	  /* assume we cannot have atoms on first page,
 | 
					
						
							|  |  |  | 	     so this must be an arity | 
					
						
							|  |  |  | 	  */ | 
					
						
							|  |  |  | 	  && reg > Yap_page_size | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 	  ) { | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | 	MarkAtomEntry(RepAtom(AtomOfTerm(reg))); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2005-05-27 21:44:00 +00:00
										 |  |  | static CELL * | 
					
						
							|  |  |  | mark_global_cell(CELL *pt) | 
					
						
							|  |  |  | {    | 
					
						
							|  |  |  |   CELL reg = *pt; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (IsVarTerm(reg)) { | 
					
						
							|  |  |  |     /* skip bitmaps */ | 
					
						
							|  |  |  |     switch(reg) { | 
					
						
							|  |  |  |     case (CELL)FunctorDouble: | 
					
						
							|  |  |  | #if SIZEOF_DOUBLE == 2*SIZEOF_LONG_INT
 | 
					
						
							|  |  |  |       return pt + 4; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  |       return pt + 3; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  |     case (CELL)FunctorBigInt: | 
					
						
							|  |  |  |       { | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  | 	Int sz = 2 + | 
					
						
							|  |  |  | 	  (sizeof(MP_INT)+ | 
					
						
							|  |  |  | 	   (((MP_INT *)(pt+1))->_mp_alloc*sizeof(mp_limb_t)))/sizeof(CELL); | 
					
						
							|  |  |  | 	return pt + sz; | 
					
						
							| 
									
										
										
										
											2005-05-27 21:44:00 +00:00
										 |  |  |       } | 
					
						
							|  |  |  |     case (CELL)FunctorLongInt: | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |       return pt + 3; | 
					
						
							| 
									
										
										
										
											2005-05-27 21:44:00 +00:00
										 |  |  |       break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } else if (IsAtomTerm(reg)) { | 
					
						
							|  |  |  |     MarkAtomEntry(RepAtom(AtomOfTerm(reg))); | 
					
						
							|  |  |  |     return pt+1; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return pt+1; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | static void | 
					
						
							|  |  |  | mark_global(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2005-05-27 21:44:00 +00:00
										 |  |  |   CELL *pt; | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   /*
 | 
					
						
							|  |  |  |    * to clean the global now that functors are just variables pointing to | 
					
						
							|  |  |  |    * the code  | 
					
						
							|  |  |  |    */ | 
					
						
							| 
									
										
										
										
											2005-05-27 21:44:00 +00:00
										 |  |  | #if COROUTINING
 | 
					
						
							| 
									
										
										
										
											2006-08-22 16:12:46 +00:00
										 |  |  |   pt = (CELL *)DelayTop(); | 
					
						
							|  |  |  | #else
 | 
					
						
							| 
									
										
										
										
											2005-05-27 21:44:00 +00:00
										 |  |  |   pt = H0; | 
					
						
							| 
									
										
										
										
											2006-08-22 16:12:46 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2005-05-27 21:44:00 +00:00
										 |  |  |   while (pt < H) { | 
					
						
							|  |  |  |     pt = mark_global_cell(pt); | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | mark_stacks(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   mark_trail(); | 
					
						
							|  |  |  |   mark_local(); | 
					
						
							|  |  |  |   mark_global(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  | mark_streams(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   int i; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   for (i=0; i < MaxStreams; i++) { | 
					
						
							|  |  |  |     if (!(Stream[i].status & (Free_Stream_f|Socket_Stream_f|InMemory_Stream_f|Pipe_Stream_f))) { | 
					
						
							|  |  |  | 	/* This is a file, so it has a name */ | 
					
						
							|  |  |  | 	AtomEntry *ae = RepAtom(Stream[i].u.file.name); | 
					
						
							|  |  |  | 	MarkAtomEntry(ae); | 
					
						
							|  |  |  | 	ae = RepAtom(AtomOfTerm(Stream[i].u.file.user_name)); | 
					
						
							|  |  |  | 	MarkAtomEntry(ae); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   for (i=0;i<NOfFileAliases;i++) { | 
					
						
							|  |  |  |     AtomEntry *ae = RepAtom(FileAliases[i].name); | 
					
						
							|  |  |  |     MarkAtomEntry(ae); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static void | 
					
						
							|  |  |  | clean_atom_list(AtomHashEntry *HashPtr) | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   Atom atm = HashPtr->Entry; | 
					
						
							|  |  |  |   Atom *patm = &(HashPtr->Entry); | 
					
						
							|  |  |  |   while (atm != NIL) { | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |     AtomEntry *at =  RepAtom(atm); | 
					
						
							|  |  |  |     if (AtomResetMark(at) || | 
					
						
							|  |  |  | 	at->PropsOfAE != NIL || | 
					
						
							|  |  |  | 	(AGCHook != NULL && !AGCHook(atm))) { | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  |       patm = &(at->NextOfAE); | 
					
						
							|  |  |  |       atm = at->NextOfAE; | 
					
						
							|  |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |       NOfAtoms--; | 
					
						
							| 
									
										
										
										
											2006-12-13 16:10:26 +00:00
										 |  |  |       if (IsWideAtom(atm)) { | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  | #ifdef DEBUG_RESTORE3
 | 
					
						
							| 
									
										
										
										
											2006-12-13 16:10:26 +00:00
										 |  |  | 	fprintf(errout, "Purged %p:%S\n", at, at->WStrOfAE); | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2006-12-13 16:10:26 +00:00
										 |  |  | 	agc_collected += sizeof(AtomEntry)+wcslen(at->WStrOfAE); | 
					
						
							|  |  |  |       } else { | 
					
						
							|  |  |  | #ifdef DEBUG_RESTORE3
 | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  | 	fprintf(stderr, "Purged %p:%s patm=%p %p\n", at, at->StrOfAE, patm, at->NextOfAE); | 
					
						
							| 
									
										
										
										
											2006-12-13 16:10:26 +00:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 	agc_collected += sizeof(AtomEntry)+strlen(at->StrOfAE); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |       *patm = atm = at->NextOfAE; | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  |       Yap_FreeCodeSpace((char *)at); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * This is the really tough part, to restore the whole of the heap  | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | static void  | 
					
						
							|  |  |  | clean_atoms(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   AtomHashEntry *HashPtr = HashChain; | 
					
						
							|  |  |  |   register int    i; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |   AtomResetMark(AtomFoundVar); | 
					
						
							|  |  |  |   AtomResetMark(AtomFreeTerm); | 
					
						
							| 
									
										
										
										
											2003-10-28 01:16:03 +00:00
										 |  |  |   for (i = 0; i < AtomHashTableSize; ++i) { | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |     clean_atom_list(HashPtr); | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  |     HashPtr++; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |   HashPtr = WideHashChain; | 
					
						
							| 
									
										
										
										
											2006-11-27 17:42:03 +00:00
										 |  |  |   for (i = 0; i < WideAtomHashTableSize; ++i) { | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |     clean_atom_list(HashPtr); | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  |     HashPtr++; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |   clean_atom_list(&INVISIBLECHAIN); | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-11 17:38:10 +00:00
										 |  |  | static void | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | atom_gc(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   int		gc_verbose = Yap_is_gc_verbose(); | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  |   int           gc_trace = 0; | 
					
						
							|  |  |  |    | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2004-03-02 16:44:58 +00:00
										 |  |  |   UInt		time_start, agc_time; | 
					
						
							| 
									
										
										
										
											2007-03-21 18:32:50 +00:00
										 |  |  |   return; | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   if (Yap_GetValue(AtomGcTrace) != TermNil) | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  |     gc_trace = 1; | 
					
						
							|  |  |  |   agc_calls++; | 
					
						
							|  |  |  |   agc_collected = 0; | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |    | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  |   if (gc_trace) { | 
					
						
							| 
									
										
										
										
											2004-06-23 17:24:20 +00:00
										 |  |  |     fprintf(Yap_stderr, "%% agc:\n"); | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  |   } else if (gc_verbose) { | 
					
						
							| 
									
										
										
										
											2004-06-23 17:24:20 +00:00
										 |  |  |     fprintf(Yap_stderr, "%%   Start of atom garbage collection %d:\n", agc_calls); | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   time_start = Yap_cputime(); | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  |   /* get the number of active registers */ | 
					
						
							|  |  |  |   YAPEnterCriticalSection(); | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |   init_reg_copies(); | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  |   mark_stacks(); | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |   mark_streams(); | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  |   mark_atoms(); | 
					
						
							|  |  |  |   clean_atoms(); | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |   AGcLastCall = NOfAtoms; | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  |   YAPLeaveCriticalSection(); | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   agc_time = Yap_cputime()-time_start; | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  |   tot_agc_time += agc_time; | 
					
						
							|  |  |  |   tot_agc_recovered += agc_collected; | 
					
						
							|  |  |  |   if (gc_verbose) { | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |     fprintf(Yap_stderr, "%%   Collected %lld bytes.\n", agc_collected); | 
					
						
							| 
									
										
										
										
											2004-06-23 17:24:20 +00:00
										 |  |  |     fprintf(Yap_stderr, "%%   GC %d took %g sec, total of %g sec doing GC so far.\n", agc_calls, (double)agc_time/1000, (double)tot_agc_time/1000); | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-11 17:38:10 +00:00
										 |  |  | void | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_atom_gc(void) | 
					
						
							| 
									
										
										
										
											2002-11-11 17:38:10 +00:00
										 |  |  | { | 
					
						
							|  |  |  |   atom_gc(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  | static Int | 
					
						
							|  |  |  | p_atom_gc(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #ifndef FIXED_STACKS
 | 
					
						
							|  |  |  |   atom_gc(); | 
					
						
							|  |  |  | #endif  /* FIXED_STACKS */
 | 
					
						
							| 
									
										
										
										
											2005-05-31 19:42:28 +00:00
										 |  |  |   return TRUE; | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static Int | 
					
						
							|  |  |  | p_inform_agc(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Term tn = MkIntegerTerm(tot_agc_time); | 
					
						
							|  |  |  |   Term tt = MkIntegerTerm(agc_calls); | 
					
						
							|  |  |  |   Term ts = MkIntegerTerm(tot_agc_recovered); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  |   return(Yap_unify(tn, ARG2) && Yap_unify(tt, ARG1) && Yap_unify(ts, ARG3)); | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  | static Int | 
					
						
							|  |  |  | p_agc_threshold(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   Term t = Deref(ARG1); | 
					
						
							|  |  |  |   if (IsVarTerm(t)) { | 
					
						
							|  |  |  |     return Yap_unify(ARG1, MkIntegerTerm(AGcThreshold)); | 
					
						
							|  |  |  |   } else if (!IsIntegerTerm(t)) { | 
					
						
							|  |  |  |     Yap_Error(TYPE_ERROR_INTEGER,t,"prolog_flag/2 agc_margin"); | 
					
						
							|  |  |  |     return FALSE; | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     Int i = IntegerOfTerm(t); | 
					
						
							|  |  |  |     if (i<0) { | 
					
						
							|  |  |  |       Yap_Error(DOMAIN_ERROR_NOT_LESS_THAN_ZERO,t,"prolog_flag/2 agc_margin"); | 
					
						
							|  |  |  |       return FALSE; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       AGcThreshold = i; | 
					
						
							|  |  |  |       return TRUE; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  | void  | 
					
						
							| 
									
										
										
										
											2002-11-18 18:18:05 +00:00
										 |  |  | Yap_init_agc(void) | 
					
						
							| 
									
										
										
										
											2002-06-05 03:59:50 +00:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2004-11-18 22:32:40 +00:00
										 |  |  |   Yap_InitCPred("$atom_gc", 0, p_atom_gc, HiddenPredFlag); | 
					
						
							|  |  |  |   Yap_InitCPred("$inform_agc", 3, p_inform_agc, HiddenPredFlag); | 
					
						
							| 
									
										
										
										
											2007-02-18 00:26:36 +00:00
										 |  |  |   Yap_InitCPred("$agc_threshold", 1, p_agc_threshold, HiddenPredFlag|SafePredFlag); | 
					
						
							| 
									
										
										
										
											2002-06-04 18:21:55 +00:00
										 |  |  | } |