| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | #include "master.hpp"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace factor | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-18 21:31:59 -04:00
										 |  |  | callback_heap::callback_heap(cell size, factor_vm *parent_) : | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | 	seg(new segment(size,true)), | 
					
						
							|  |  |  | 	here(seg->start), | 
					
						
							| 
									
										
										
										
											2009-10-18 21:31:59 -04:00
										 |  |  | 	parent(parent_) {} | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | callback_heap::~callback_heap() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	delete seg; | 
					
						
							|  |  |  | 	seg = NULL; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void factor_vm::init_callbacks(cell size) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	callbacks = new callback_heap(size,this); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-04 17:46:36 -04:00
										 |  |  | bool callback_heap::setup_seh_p() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if defined(WINDOWS) && defined(FACTOR_X86)
 | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool callback_heap::return_takes_param_p() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | #if defined(FACTOR_X86) || defined(FACTOR_AMD64)
 | 
					
						
							|  |  |  | 	return true; | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | 	return false; | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | instruction_operand callback_heap::callback_operand(code_block *stub, cell index) | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-23 01:33:53 -04:00
										 |  |  | 	tagged<array> code_template(parent->special_objects[CALLBACK_STUB]); | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-10 07:20:32 -05:00
										 |  |  | 	cell rel_class = untag_fixnum(array_nth(code_template.untagged(),3 * index + 1)); | 
					
						
							|  |  |  | 	cell rel_type  = untag_fixnum(array_nth(code_template.untagged(),3 * index + 2)); | 
					
						
							|  |  |  | 	cell offset    = untag_fixnum(array_nth(code_template.untagged(),3 * index + 3)); | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-02 01:48:41 -05:00
										 |  |  | 	relocation_entry rel( | 
					
						
							|  |  |  | 		(relocation_type)rel_type, | 
					
						
							|  |  |  | 		(relocation_class)rel_class, | 
					
						
							|  |  |  | 		offset); | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-02 01:48:41 -05:00
										 |  |  | 	instruction_operand op(rel,stub,0); | 
					
						
							| 
									
										
										
										
											2010-04-04 17:46:36 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return op; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void callback_heap::store_callback_operand(code_block *stub, cell index) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	parent->store_external_address(callback_operand(stub,index)); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void callback_heap::store_callback_operand(code_block *stub, cell index, cell value) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	callback_operand(stub,index).store_value(value); | 
					
						
							| 
									
										
										
										
											2010-01-10 07:20:32 -05:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-12-02 01:48:41 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-10 07:20:32 -05:00
										 |  |  | void callback_heap::update(code_block *stub) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-04-04 17:46:36 -04:00
										 |  |  | 	store_callback_operand(stub,setup_seh_p() ? 2 : 1,(cell)callback_entry_point(stub)); | 
					
						
							| 
									
										
										
										
											2009-12-02 01:48:41 -05:00
										 |  |  | 	stub->flush_icache(); | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-02 07:03:30 -05:00
										 |  |  | code_block *callback_heap::add(cell owner, cell return_rewind) | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-23 01:33:53 -04:00
										 |  |  | 	tagged<array> code_template(parent->special_objects[CALLBACK_STUB]); | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | 	tagged<byte_array> insns(array_nth(code_template.untagged(),0)); | 
					
						
							|  |  |  | 	cell size = array_capacity(insns.untagged()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-02 01:48:41 -05:00
										 |  |  | 	cell bump = align(size + sizeof(code_block),data_alignment); | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | 	if(here + bump > seg->end) fatal_error("Out of callback space",0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-02 01:48:41 -05:00
										 |  |  | 	free_heap_block *free_block = (free_heap_block *)here; | 
					
						
							|  |  |  | 	free_block->make_free(bump); | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | 	here += bump; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-02 01:48:41 -05:00
										 |  |  | 	code_block *stub = (code_block *)free_block; | 
					
						
							|  |  |  | 	stub->owner = owner; | 
					
						
							| 
									
										
										
										
											2009-12-02 05:28:15 -05:00
										 |  |  | 	stub->parameters = false_object; | 
					
						
							| 
									
										
										
										
											2009-12-02 01:48:41 -05:00
										 |  |  | 	stub->relocation = false_object; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-18 02:51:27 -05:00
										 |  |  | 	memcpy(stub->entry_point(),insns->data<void>(),size); | 
					
						
							| 
									
										
										
										
											2010-01-02 07:03:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-10 07:20:32 -05:00
										 |  |  | 	/* Store VM pointer */ | 
					
						
							|  |  |  | 	store_callback_operand(stub,0,(cell)parent); | 
					
						
							| 
									
										
										
										
											2010-04-03 20:24:33 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-04-04 17:46:36 -04:00
										 |  |  | 	cell index; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	if(setup_seh_p()) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		store_callback_operand(stub,1); | 
					
						
							|  |  |  | 		index = 1; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	else | 
					
						
							|  |  |  | 		index = 0; | 
					
						
							| 
									
										
										
										
											2010-04-03 20:24:33 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	/* Store VM pointer */ | 
					
						
							|  |  |  | 	store_callback_operand(stub,index + 2,(cell)parent); | 
					
						
							| 
									
										
										
										
											2010-01-10 07:20:32 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-02 07:03:30 -05:00
										 |  |  | 	/* On x86, the RET instruction takes an argument which depends on
 | 
					
						
							|  |  |  | 	the callback's calling convention */ | 
					
						
							| 
									
										
										
										
											2010-04-04 17:46:36 -04:00
										 |  |  | 	if(return_takes_param_p()) | 
					
						
							|  |  |  | 		store_callback_operand(stub,index + 3,return_rewind); | 
					
						
							| 
									
										
										
										
											2010-01-02 07:03:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | 	update(stub); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return stub; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-02 04:31:57 -05:00
										 |  |  | struct callback_updater { | 
					
						
							|  |  |  | 	callback_heap *callbacks; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	explicit callback_updater(callback_heap *callbacks_) : callbacks(callbacks_) {} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	void operator()(code_block *stub) | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		callbacks->update(stub); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void callback_heap::update() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	callback_updater updater(this); | 
					
						
							|  |  |  | 	each_callback(updater); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | void factor_vm::primitive_callback() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-01-02 07:03:30 -05:00
										 |  |  | 	cell return_rewind = to_cell(ctx->pop()); | 
					
						
							| 
									
										
										
										
											2009-12-18 16:59:56 -05:00
										 |  |  | 	tagged<word> w(ctx->pop()); | 
					
						
							| 
									
										
										
										
											2010-01-02 07:03:30 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | 	w.untag_check(this); | 
					
						
							| 
									
										
										
										
											2011-05-20 18:11:50 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	void* func = callbacks->add(w.value(),return_rewind)->entry_point(); | 
					
						
							|  |  |  | 	CODE_TO_FUNCTION_POINTER_CALLBACK(this, func); | 
					
						
							|  |  |  | 	ctx->push(allot_alien(func)); | 
					
						
							| 
									
										
										
										
											2009-10-16 12:39:22 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } |