| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | #include "master.hpp"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | namespace factor { | 
					
						
							| 
									
										
										
										
											2009-05-04 02:46:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | context::context(cell datastack_size, cell retainstack_size, | 
					
						
							|  |  |  |                  cell callstack_size) | 
					
						
							| 
									
										
										
										
											2015-01-05 06:59:54 -05:00
										 |  |  |     : callstack_top(0), | 
					
						
							|  |  |  |       callstack_bottom(0), | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |       datastack(0), | 
					
						
							|  |  |  |       retainstack(0), | 
					
						
							|  |  |  |       callstack_save(0), | 
					
						
							|  |  |  |       datastack_seg(new segment(datastack_size, false)), | 
					
						
							|  |  |  |       retainstack_seg(new segment(retainstack_size, false)), | 
					
						
							|  |  |  |       callstack_seg(new segment(callstack_size, false)) { | 
					
						
							|  |  |  |   reset(); | 
					
						
							| 
									
										
										
										
											2010-03-18 05:06:00 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void context::reset_datastack() { | 
					
						
							|  |  |  |   datastack = datastack_seg->start - sizeof(cell); | 
					
						
							| 
									
										
										
										
											2015-05-22 10:52:13 -04:00
										 |  |  |   fill_stack_seg(datastack, datastack_seg, 0x11111111); | 
					
						
							| 
									
										
										
										
											2010-03-18 05:06:00 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void context::reset_retainstack() { | 
					
						
							|  |  |  |   retainstack = retainstack_seg->start - sizeof(cell); | 
					
						
							| 
									
										
										
										
											2015-05-22 10:52:13 -04:00
										 |  |  |   fill_stack_seg(retainstack, retainstack_seg, 0x22222222); | 
					
						
							| 
									
										
										
										
											2010-03-26 22:44:43 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void context::reset_callstack() { | 
					
						
							|  |  |  |   callstack_top = callstack_bottom = CALLSTACK_BOTTOM(this); | 
					
						
							| 
									
										
										
										
											2010-03-18 05:06:00 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void context::reset_context_objects() { | 
					
						
							|  |  |  |   memset_cell(context_objects, false_object, | 
					
						
							|  |  |  |               context_object_count * sizeof(cell)); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-22 10:52:13 -04:00
										 |  |  | void context::fill_stack_seg(cell top_ptr, segment* seg, cell pattern) { | 
					
						
							|  |  |  | #ifdef FACTOR_DEBUG
 | 
					
						
							|  |  |  |   cell clear_start = top_ptr + sizeof(cell); | 
					
						
							|  |  |  |   cell clear_size = seg->end - clear_start; | 
					
						
							|  |  |  |   memset_cell((void*)clear_start, pattern, clear_size); | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void context::reset() { | 
					
						
							|  |  |  |   reset_datastack(); | 
					
						
							|  |  |  |   reset_retainstack(); | 
					
						
							|  |  |  |   reset_callstack(); | 
					
						
							|  |  |  |   reset_context_objects(); | 
					
						
							| 
									
										
										
										
											2010-03-26 22:44:43 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-12-04 14:45:13 -05:00
										 |  |  | void context::fix_stacks() { | 
					
						
							|  |  |  |   if (datastack + sizeof(cell) < datastack_seg->start || | 
					
						
							|  |  |  |       datastack + stack_reserved >= datastack_seg->end) | 
					
						
							|  |  |  |     reset_datastack(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (retainstack + sizeof(cell) < retainstack_seg->start || | 
					
						
							|  |  |  |       retainstack + stack_reserved >= retainstack_seg->end) | 
					
						
							|  |  |  |     reset_retainstack(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | context::~context() { | 
					
						
							|  |  |  |   delete datastack_seg; | 
					
						
							|  |  |  |   delete retainstack_seg; | 
					
						
							|  |  |  |   delete callstack_seg; | 
					
						
							| 
									
										
										
										
											2010-03-26 22:44:43 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* called on startup */ | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  | /* Allocates memory (new_context()) */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::init_contexts(cell datastack_size_, cell retainstack_size_, | 
					
						
							|  |  |  |                               cell callstack_size_) { | 
					
						
							|  |  |  |   datastack_size = datastack_size_; | 
					
						
							|  |  |  |   retainstack_size = retainstack_size_; | 
					
						
							|  |  |  |   callstack_size = callstack_size_; | 
					
						
							| 
									
										
										
										
											2010-03-26 22:44:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   ctx = NULL; | 
					
						
							|  |  |  |   spare_ctx = new_context(); | 
					
						
							| 
									
										
										
										
											2010-03-26 22:44:43 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::delete_contexts() { | 
					
						
							|  |  |  |   FACTOR_ASSERT(!ctx); | 
					
						
							| 
									
										
										
										
											2015-05-29 06:25:12 -04:00
										 |  |  |   FACTOR_FOR_EACH(unused_contexts) { | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |     delete *iter; | 
					
						
							|  |  |  |     iter++; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-03-26 22:44:43 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | context* factor_vm::new_context() { | 
					
						
							|  |  |  |   context* new_context; | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   if (unused_contexts.empty()) { | 
					
						
							|  |  |  |     new_context = new context(datastack_size, retainstack_size, callstack_size); | 
					
						
							|  |  |  |   } else { | 
					
						
							|  |  |  |     new_context = unused_contexts.back(); | 
					
						
							|  |  |  |     unused_contexts.pop_back(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2010-03-26 22:44:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   new_context->reset(); | 
					
						
							| 
									
										
										
										
											2010-03-26 22:44:43 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   active_contexts.insert(new_context); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   return new_context; | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:08 -04:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::init_context(context* ctx) { | 
					
						
							|  |  |  |   ctx->context_objects[OBJ_CONTEXT] = allot_alien(ctx); | 
					
						
							| 
									
										
										
										
											2010-04-01 22:12:45 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  | /* Allocates memory (init_context(), but not parent->new_context() */ | 
					
						
							| 
									
										
										
										
											2015-05-18 12:43:20 -04:00
										 |  |  | VM_C_API context* new_context(factor_vm* parent) { | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   context* new_context = parent->new_context(); | 
					
						
							|  |  |  |   parent->init_context(new_context); | 
					
						
							|  |  |  |   return new_context; | 
					
						
							| 
									
										
										
										
											2010-03-27 02:55:49 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-18 12:43:20 -04:00
										 |  |  | void factor_vm::delete_context() { | 
					
						
							|  |  |  |   unused_contexts.push_back(ctx); | 
					
						
							|  |  |  |   active_contexts.erase(ctx); | 
					
						
							| 
									
										
										
										
											2010-09-17 23:52:27 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   while (unused_contexts.size() > 10) { | 
					
						
							|  |  |  |     context* stale_context = unused_contexts.front(); | 
					
						
							|  |  |  |     unused_contexts.pop_front(); | 
					
						
							|  |  |  |     delete stale_context; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-18 12:43:20 -04:00
										 |  |  | VM_C_API void delete_context(factor_vm* parent) { | 
					
						
							|  |  |  |   parent->delete_context(); | 
					
						
							| 
									
										
										
										
											2010-03-30 21:47:48 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  | /* Allocates memory (init_context()) */ | 
					
						
							| 
									
										
										
										
											2015-05-18 12:43:20 -04:00
										 |  |  | VM_C_API void reset_context(factor_vm* parent) { | 
					
						
							| 
									
										
										
										
											2015-05-22 09:29:38 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   // The function is used by (start-context-and-delete) which expects
 | 
					
						
							|  |  |  |   // the top two datastack items to be preserved after the context has
 | 
					
						
							|  |  |  |   // been resetted.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   context* ctx = parent->ctx; | 
					
						
							|  |  |  |   cell arg1 = ctx->pop(); | 
					
						
							|  |  |  |   cell arg2 = ctx->pop(); | 
					
						
							|  |  |  |   ctx->reset(); | 
					
						
							|  |  |  |   ctx->push(arg2); | 
					
						
							|  |  |  |   ctx->push(arg1); | 
					
						
							|  |  |  |   parent->init_context(ctx); | 
					
						
							| 
									
										
										
										
											2010-08-02 20:28:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:08 -04:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | cell factor_vm::begin_callback(cell quot_) { | 
					
						
							|  |  |  |   data_root<object> quot(quot_, this); | 
					
						
							| 
									
										
										
										
											2010-04-01 22:12:45 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   ctx->reset(); | 
					
						
							|  |  |  |   spare_ctx = new_context(); | 
					
						
							|  |  |  |   callback_ids.push_back(callback_id++); | 
					
						
							| 
									
										
										
										
											2010-04-01 22:12:45 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   init_context(ctx); | 
					
						
							| 
									
										
										
										
											2010-04-01 22:12:45 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   return quot.value(); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | cell begin_callback(factor_vm* parent, cell quot) { | 
					
						
							|  |  |  |   return parent->begin_callback(quot); | 
					
						
							| 
									
										
										
										
											2009-08-17 16:37:04 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::end_callback() { | 
					
						
							|  |  |  |   callback_ids.pop_back(); | 
					
						
							| 
									
										
										
										
											2015-05-18 12:43:20 -04:00
										 |  |  |   delete_context(); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void end_callback(factor_vm* parent) { parent->end_callback(); } | 
					
						
							| 
									
										
										
										
											2009-08-17 16:37:04 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::primitive_current_callback() { | 
					
						
							|  |  |  |   ctx->push(tag_fixnum(callback_ids.back())); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::primitive_context_object() { | 
					
						
							|  |  |  |   fixnum n = untag_fixnum(ctx->peek()); | 
					
						
							|  |  |  |   ctx->replace(ctx->context_objects[n]); | 
					
						
							| 
									
										
										
										
											2010-03-18 05:06:00 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::primitive_set_context_object() { | 
					
						
							|  |  |  |   fixnum n = untag_fixnum(ctx->pop()); | 
					
						
							|  |  |  |   cell value = ctx->pop(); | 
					
						
							|  |  |  |   ctx->context_objects[n] = value; | 
					
						
							| 
									
										
										
										
											2010-03-18 05:06:00 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::primitive_context_object_for() { | 
					
						
							|  |  |  |   context* other_ctx = (context*)pinned_alien_offset(ctx->pop()); | 
					
						
							|  |  |  |   fixnum n = untag_fixnum(ctx->peek()); | 
					
						
							|  |  |  |   ctx->replace(other_ctx->context_objects[n]); | 
					
						
							| 
									
										
										
										
											2010-03-29 20:40:17 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-16 01:00:08 -04:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2015-02-28 14:41:32 -05:00
										 |  |  | cell factor_vm::stack_to_array(cell bottom, cell top, vm_error_type error) { | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   fixnum depth = (fixnum)(top - bottom + sizeof(cell)); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-02-28 14:41:32 -05:00
										 |  |  |   if (depth < 0) { | 
					
						
							|  |  |  |     general_error(error, false_object, false_object); | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-02-28 14:41:32 -05:00
										 |  |  |   array* a = allot_uninitialized_array<array>(depth / sizeof(cell)); | 
					
						
							|  |  |  |   memcpy(a + 1, (void*)bottom, depth); | 
					
						
							|  |  |  |   return tag<array>(a); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | cell factor_vm::datastack_to_array(context* ctx) { | 
					
						
							| 
									
										
										
										
											2015-02-28 14:41:32 -05:00
										 |  |  |   return stack_to_array(ctx->datastack_seg->start, | 
					
						
							|  |  |  |                         ctx->datastack, | 
					
						
							|  |  |  |                         ERROR_DATASTACK_UNDERFLOW); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::primitive_datastack() { ctx->push(datastack_to_array(ctx)); } | 
					
						
							| 
									
										
										
										
											2010-03-29 20:40:17 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::primitive_datastack_for() { | 
					
						
							|  |  |  |   context* other_ctx = (context*)pinned_alien_offset(ctx->peek()); | 
					
						
							|  |  |  |   ctx->replace(datastack_to_array(other_ctx)); | 
					
						
							| 
									
										
										
										
											2010-03-29 20:40:17 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | cell factor_vm::retainstack_to_array(context* ctx) { | 
					
						
							| 
									
										
										
										
											2015-02-28 14:41:32 -05:00
										 |  |  |   return stack_to_array(ctx->retainstack_seg->start, | 
					
						
							|  |  |  |                         ctx->retainstack, | 
					
						
							|  |  |  |                         ERROR_RETAINSTACK_UNDERFLOW); | 
					
						
							| 
									
										
										
										
											2010-03-29 20:40:17 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::primitive_retainstack() { | 
					
						
							|  |  |  |   ctx->push(retainstack_to_array(ctx)); | 
					
						
							| 
									
										
										
										
											2010-03-29 20:40:17 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::primitive_retainstack_for() { | 
					
						
							|  |  |  |   context* other_ctx = (context*)pinned_alien_offset(ctx->peek()); | 
					
						
							|  |  |  |   ctx->replace(retainstack_to_array(other_ctx)); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* returns pointer to top of stack */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | cell factor_vm::array_to_stack(array* array, cell bottom) { | 
					
						
							|  |  |  |   cell depth = array_capacity(array) * sizeof(cell); | 
					
						
							|  |  |  |   memcpy((void*)bottom, array + 1, depth); | 
					
						
							|  |  |  |   return bottom + depth - sizeof(cell); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::set_datastack(context* ctx, array* array) { | 
					
						
							|  |  |  |   ctx->datastack = array_to_stack(array, ctx->datastack_seg->start); | 
					
						
							| 
									
										
										
										
											2010-03-29 20:40:17 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::primitive_set_datastack() { | 
					
						
							|  |  |  |   set_datastack(ctx, untag_check<array>(ctx->pop())); | 
					
						
							| 
									
										
										
										
											2010-03-29 20:40:17 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::set_retainstack(context* ctx, array* array) { | 
					
						
							|  |  |  |   ctx->retainstack = array_to_stack(array, ctx->retainstack_seg->start); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::primitive_set_retainstack() { | 
					
						
							|  |  |  |   set_retainstack(ctx, untag_check<array>(ctx->pop())); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /* Used to implement call( */ | 
					
						
							| 
									
										
										
										
											2013-05-11 21:53:18 -04:00
										 |  |  | void factor_vm::primitive_check_datastack() { | 
					
						
							|  |  |  |   fixnum out = to_fixnum(ctx->pop()); | 
					
						
							|  |  |  |   fixnum in = to_fixnum(ctx->pop()); | 
					
						
							|  |  |  |   fixnum height = out - in; | 
					
						
							|  |  |  |   array* saved_datastack = untag_check<array>(ctx->pop()); | 
					
						
							|  |  |  |   fixnum saved_height = array_capacity(saved_datastack); | 
					
						
							|  |  |  |   fixnum current_height = | 
					
						
							|  |  |  |       (ctx->datastack - ctx->datastack_seg->start + sizeof(cell)) / | 
					
						
							|  |  |  |       sizeof(cell); | 
					
						
							|  |  |  |   if (current_height - height != saved_height) | 
					
						
							|  |  |  |     ctx->push(false_object); | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     cell* ds_bot = (cell*)ctx->datastack_seg->start; | 
					
						
							|  |  |  |     for (fixnum i = 0; i < saved_height - in; i++) { | 
					
						
							|  |  |  |       if (ds_bot[i] != array_nth(saved_datastack, i)) { | 
					
						
							|  |  |  |         ctx->push(false_object); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ctx->push(true_object); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void factor_vm::primitive_load_locals() { | 
					
						
							|  |  |  |   fixnum count = untag_fixnum(ctx->pop()); | 
					
						
							|  |  |  |   memcpy((cell*)(ctx->retainstack + sizeof(cell)), | 
					
						
							|  |  |  |          (cell*)(ctx->datastack - sizeof(cell) * (count - 1)), | 
					
						
							|  |  |  |          sizeof(cell) * count); | 
					
						
							|  |  |  |   ctx->datastack -= sizeof(cell) * count; | 
					
						
							|  |  |  |   ctx->retainstack += sizeof(cell) * count; | 
					
						
							| 
									
										
										
										
											2009-11-05 20:03:51 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-04 02:46:13 -04:00
										 |  |  | } |