| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | #include "master.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | void reset_datastack(void) | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | 	ds = ds_bot - CELLS; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | void reset_retainstack(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	rs = rs_bot - CELLS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #define RESERVED (64 * CELLS)
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void fix_stacks(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	if(ds + CELLS < ds_bot || ds + RESERVED >= ds_top) reset_datastack(); | 
					
						
							|  |  |  | 	if(rs + CELLS < rs_bot || rs + RESERVED >= rs_top) reset_retainstack(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* called before entry into foreign C code. Note that ds and rs might
 | 
					
						
							|  |  |  | be stored in registers, so callbacks must save and restore the correct values */ | 
					
						
							|  |  |  | void save_stacks(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-04-02 19:50:35 -04:00
										 |  |  | 	if(stack_chain) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		stack_chain->datastack = ds; | 
					
						
							|  |  |  | 		stack_chain->retainstack = rs; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-22 00:13:14 -05:00
										 |  |  | F_CONTEXT *alloc_context(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	F_CONTEXT *context; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(unused_contexts) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		context = unused_contexts; | 
					
						
							|  |  |  | 		unused_contexts = unused_contexts->next; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		context = safe_malloc(sizeof(F_CONTEXT)); | 
					
						
							|  |  |  | 		context->datastack_region = alloc_segment(ds_size); | 
					
						
							|  |  |  | 		context->retainstack_region = alloc_segment(rs_size); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return context; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void dealloc_context(F_CONTEXT *context) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	context->next = unused_contexts; | 
					
						
							|  |  |  | 	unused_contexts = context; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | /* called on entry into a compiled callback */ | 
					
						
							|  |  |  | void nest_stacks(void) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2008-11-22 00:13:14 -05:00
										 |  |  | 	F_CONTEXT *new_stacks = alloc_context(); | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	new_stacks->callstack_bottom = (F_STACK_FRAME *)-1; | 
					
						
							|  |  |  | 	new_stacks->callstack_top = (F_STACK_FRAME *)-1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* note that these register values are not necessarily valid stack
 | 
					
						
							|  |  |  | 	pointers. they are merely saved non-volatile registers, and are | 
					
						
							|  |  |  | 	restored in unnest_stacks(). consider this scenario: | 
					
						
							|  |  |  | 	- factor code calls C function | 
					
						
							|  |  |  | 	- C function saves ds/cs registers (since they're non-volatile) | 
					
						
							|  |  |  | 	- C function clobbers them | 
					
						
							|  |  |  | 	- C function calls Factor callback | 
					
						
							|  |  |  | 	- Factor callback returns | 
					
						
							|  |  |  | 	- C function restores registers | 
					
						
							|  |  |  | 	- C function returns to Factor code */ | 
					
						
							|  |  |  | 	new_stacks->datastack_save = ds; | 
					
						
							|  |  |  | 	new_stacks->retainstack_save = rs; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* save per-callback userenv */ | 
					
						
							|  |  |  | 	new_stacks->current_callback_save = userenv[CURRENT_CALLBACK_ENV]; | 
					
						
							|  |  |  | 	new_stacks->catchstack_save = userenv[CATCHSTACK_ENV]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	new_stacks->next = stack_chain; | 
					
						
							|  |  |  | 	stack_chain = new_stacks; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	reset_datastack(); | 
					
						
							|  |  |  | 	reset_retainstack(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* called when leaving a compiled callback */ | 
					
						
							|  |  |  | void unnest_stacks(void) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ds = stack_chain->datastack_save; | 
					
						
							|  |  |  | 	rs = stack_chain->retainstack_save; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	/* restore per-callback userenv */ | 
					
						
							|  |  |  | 	userenv[CURRENT_CALLBACK_ENV] = stack_chain->current_callback_save; | 
					
						
							|  |  |  | 	userenv[CATCHSTACK_ENV] = stack_chain->catchstack_save; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	F_CONTEXT *old_stacks = stack_chain; | 
					
						
							|  |  |  | 	stack_chain = old_stacks->next; | 
					
						
							| 
									
										
										
										
											2008-11-22 00:13:14 -05:00
										 |  |  | 	dealloc_context(old_stacks); | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* called on startup */ | 
					
						
							|  |  |  | void init_stacks(CELL ds_size_, CELL rs_size_) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	ds_size = ds_size_; | 
					
						
							|  |  |  | 	rs_size = rs_size_; | 
					
						
							|  |  |  | 	stack_chain = NULL; | 
					
						
							| 
									
										
										
										
											2008-11-22 00:13:14 -05:00
										 |  |  | 	unused_contexts = NULL; | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-03 17:11:52 -04:00
										 |  |  | bool stack_to_array(CELL bottom, CELL top) | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	F_FIXNUM depth = (F_FIXNUM)(top - bottom + CELLS); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2007-10-03 17:11:52 -04:00
										 |  |  | 	if(depth < 0) | 
					
						
							|  |  |  | 		return false; | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		F_ARRAY *a = allot_array_internal(ARRAY_TYPE,depth / CELLS); | 
					
						
							|  |  |  | 		memcpy(a + 1,(void*)bottom,depth); | 
					
						
							|  |  |  | 		dpush(tag_object(a)); | 
					
						
							|  |  |  | 		return true; | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_datastack(void) | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2007-10-03 17:11:52 -04:00
										 |  |  | 	if(!stack_to_array(ds_bot,ds)) | 
					
						
							|  |  |  | 		general_error(ERROR_DS_UNDERFLOW,F,F,NULL); | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_retainstack(void) | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2007-10-03 17:11:52 -04:00
										 |  |  | 	if(!stack_to_array(rs_bot,rs)) | 
					
						
							|  |  |  | 		general_error(ERROR_RS_UNDERFLOW,F,F,NULL); | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* returns pointer to top of stack */ | 
					
						
							|  |  |  | CELL array_to_stack(F_ARRAY *array, CELL bottom) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	CELL depth = array_capacity(array) * CELLS; | 
					
						
							|  |  |  | 	memcpy((void*)bottom,array + 1,depth); | 
					
						
							|  |  |  | 	return bottom + depth - CELLS; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_set_datastack(void) | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	ds = array_to_stack(untag_array(dpop()),ds_bot); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_set_retainstack(void) | 
					
						
							| 
									
										
										
										
											2007-10-02 17:53:05 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	rs = array_to_stack(untag_array(dpop()),rs_bot); | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_getenv(void) | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	F_FIXNUM e = untag_fixnum_fast(dpeek()); | 
					
						
							|  |  |  | 	drepl(userenv[e]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_setenv(void) | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	F_FIXNUM e = untag_fixnum_fast(dpop()); | 
					
						
							|  |  |  | 	CELL value = dpop(); | 
					
						
							|  |  |  | 	userenv[e] = value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_exit(void) | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	exit(to_fixnum(dpop())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_millis(void) | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	box_unsigned_8(current_millis()); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_sleep(void) | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	sleep_millis(to_cell(dpop())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-11-13 04:20:34 -05:00
										 |  |  | void primitive_set_slot(void) | 
					
						
							| 
									
										
										
										
											2007-09-20 18:09:08 -04:00
										 |  |  | { | 
					
						
							|  |  |  | 	F_FIXNUM slot = untag_fixnum_fast(dpop()); | 
					
						
							|  |  |  | 	CELL obj = dpop(); | 
					
						
							|  |  |  | 	CELL value = dpop(); | 
					
						
							|  |  |  | 	set_slot(obj,slot,value); | 
					
						
							|  |  |  | } |