| 
									
										
										
										
											2008-04-05 05:01:46 -04:00
										 |  |  | void gc(void); | 
					
						
							| 
									
										
										
										
											2008-04-19 05:52:34 -04:00
										 |  |  | DLLEXPORT void minor_gc(void); | 
					
						
							| 
									
										
										
										
											2008-04-05 05:01:46 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | /* used during garbage collection only */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-25 00:39:00 -05:00
										 |  |  | F_ZONE *newspace; | 
					
						
							|  |  |  | bool performing_gc; | 
					
						
							| 
									
										
										
										
											2009-04-17 21:38:55 -04:00
										 |  |  | bool performing_compaction; | 
					
						
							| 
									
										
										
										
											2009-01-25 00:39:00 -05:00
										 |  |  | CELL collecting_gen; | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-25 00:39:00 -05:00
										 |  |  | /* if true, we collecting AGING space for the second time, so if it is still
 | 
					
						
							|  |  |  | full, we go on to collect TENURED */ | 
					
						
							|  |  |  | bool collecting_aging_again; | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-25 00:39:00 -05:00
										 |  |  | /* in case a generation fills up in the middle of a gc, we jump back
 | 
					
						
							|  |  |  | up to try collecting the next generation. */ | 
					
						
							|  |  |  | jmp_buf gc_jmp; | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | /* statistics */ | 
					
						
							| 
									
										
										
										
											2008-05-08 00:09:18 -04:00
										 |  |  | typedef struct { | 
					
						
							|  |  |  | 	CELL collections; | 
					
						
							| 
									
										
										
										
											2008-11-19 02:50:05 -05:00
										 |  |  | 	u64 gc_time; | 
					
						
							|  |  |  | 	u64 max_gc_time; | 
					
						
							| 
									
										
										
										
											2008-05-08 00:09:18 -04:00
										 |  |  | 	CELL object_count; | 
					
						
							|  |  |  | 	u64 bytes_copied; | 
					
						
							|  |  |  | } F_GC_STATS; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | F_GC_STATS gc_stats[MAX_GEN_COUNT]; | 
					
						
							| 
									
										
										
										
											2008-05-07 22:39:20 -04:00
										 |  |  | u64 cards_scanned; | 
					
						
							|  |  |  | u64 decks_scanned; | 
					
						
							| 
									
										
										
										
											2008-05-07 18:42:41 -04:00
										 |  |  | CELL code_heap_scans; | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-25 00:39:00 -05:00
										 |  |  | /* What generation was being collected when copy_code_heap_roots() was last
 | 
					
						
							| 
									
										
										
										
											2009-03-19 04:45:37 -04:00
										 |  |  | called? Until the next call to add_code_block(), future | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | collections of younger generations don't have to touch the code | 
					
						
							|  |  |  | heap. */ | 
					
						
							|  |  |  | CELL last_code_heap_scan; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* sometimes we grow the heap */ | 
					
						
							|  |  |  | bool growing_data_heap; | 
					
						
							|  |  |  | F_DATA_HEAP *old_data_heap; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-25 00:39:00 -05:00
										 |  |  | INLINE bool collecting_accumulation_gen_p(void) | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-01-25 00:39:00 -05:00
										 |  |  | 	return ((HAVE_AGING_P | 
					
						
							|  |  |  | 		&& collecting_gen == AGING | 
					
						
							|  |  |  | 		&& !collecting_aging_again) | 
					
						
							|  |  |  | 		|| collecting_gen == TENURED); | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-05 05:01:46 -04:00
										 |  |  | /* test if the pointer is in generation being collected, or a younger one. */ | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | INLINE bool should_copy(CELL untagged) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if(in_zone(newspace,untagged)) | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	if(collecting_gen == TENURED) | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	else if(HAVE_AGING_P && collecting_gen == AGING) | 
					
						
							|  |  |  | 		return !in_zone(&data_heap->generations[TENURED],untagged); | 
					
						
							|  |  |  | 	else if(HAVE_NURSERY_P && collecting_gen == NURSERY) | 
					
						
							| 
									
										
										
										
											2008-04-19 05:52:34 -04:00
										 |  |  | 		return in_zone(&nursery,untagged); | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		critical_error("Bug in should_copy",untagged); | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void copy_handle(CELL *handle); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void garbage_collection(volatile CELL gen, | 
					
						
							|  |  |  | 	bool growing_data_heap_, | 
					
						
							|  |  |  | 	CELL requested_bytes); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* We leave this many bytes free at the top of the nursery so that inline
 | 
					
						
							|  |  |  | allocation (which does not call GC because of possible roots in volatile | 
					
						
							|  |  |  | registers) does not run out of memory */ | 
					
						
							|  |  |  | #define ALLOT_BUFFER_ZONE 1024
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-05 05:01:46 -04:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  * It is up to the caller to fill in the object's fields in a meaningful | 
					
						
							|  |  |  |  * fashion! | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2009-01-25 00:39:00 -05:00
										 |  |  | INLINE void *allot_object(CELL type, CELL a) | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2008-04-05 05:01:46 -04:00
										 |  |  | 	CELL *object; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-19 05:52:34 -04:00
										 |  |  | 	if(HAVE_NURSERY_P && nursery.size - ALLOT_BUFFER_ZONE > a) | 
					
						
							| 
									
										
										
										
											2008-04-05 05:01:46 -04:00
										 |  |  | 	{ | 
					
						
							|  |  |  | 		/* If there is insufficient room, collect the nursery */ | 
					
						
							| 
									
										
										
										
											2008-04-19 05:52:34 -04:00
										 |  |  | 		if(nursery.here + ALLOT_BUFFER_ZONE + a > nursery.end) | 
					
						
							| 
									
										
										
										
											2008-04-05 05:01:46 -04:00
										 |  |  | 			garbage_collection(NURSERY,false,0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-19 05:52:34 -04:00
										 |  |  | 		CELL h = nursery.here; | 
					
						
							|  |  |  | 		nursery.here = h + align8(a); | 
					
						
							|  |  |  | 		object = (void*)h; | 
					
						
							| 
									
										
										
										
											2008-04-05 05:01:46 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2008-04-07 16:45:08 -04:00
										 |  |  | 	/* If the object is bigger than the nursery, allocate it in
 | 
					
						
							|  |  |  | 	tenured space */ | 
					
						
							| 
									
										
										
										
											2008-04-05 03:08:37 -04:00
										 |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							| 
									
										
										
										
											2008-04-05 05:01:46 -04:00
										 |  |  | 		F_ZONE *tenured = &data_heap->generations[TENURED]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* If tenured space does not have enough room, collect */ | 
					
						
							|  |  |  | 		if(tenured->here + a > tenured->end) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			gc(); | 
					
						
							|  |  |  | 			tenured = &data_heap->generations[TENURED]; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		/* If it still won't fit, grow the heap */ | 
					
						
							|  |  |  | 		if(tenured->here + a > tenured->end) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			garbage_collection(TENURED,true,a); | 
					
						
							|  |  |  | 			tenured = &data_heap->generations[TENURED]; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		object = allot_zone(tenured,a); | 
					
						
							| 
									
										
										
										
											2008-04-05 03:08:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-05 05:01:46 -04:00
										 |  |  | 		/* We have to do this */ | 
					
						
							|  |  |  | 		allot_barrier((CELL)object); | 
					
						
							| 
									
										
										
										
											2008-04-05 03:08:37 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-04-05 05:01:46 -04:00
										 |  |  | 		/* Allows initialization code to store old->new pointers
 | 
					
						
							|  |  |  | 		without hitting the write barrier in the common case of | 
					
						
							|  |  |  | 		a nursery allocation */ | 
					
						
							|  |  |  | 		write_barrier((CELL)object); | 
					
						
							| 
									
										
										
										
											2008-04-05 03:08:37 -04:00
										 |  |  | 	} | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	*object = tag_header(type); | 
					
						
							|  |  |  | 	return object; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-24 18:01:01 -05:00
										 |  |  | void copy_reachable_objects(CELL scan, CELL *end); | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_gc(void); | 
					
						
							|  |  |  | void primitive_gc_stats(void); | 
					
						
							| 
									
										
										
										
											2009-01-25 00:39:00 -05:00
										 |  |  | void clear_gc_stats(void); | 
					
						
							|  |  |  | void primitive_clear_gc_stats(void); | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_become(void); |