| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | #include "master.hpp"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | namespace factor { | 
					
						
							| 
									
										
										
										
											2009-05-04 02:46:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | /* Simple non-optimizing compiler.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | This is one of the two compilers implementing Factor; the second one is written | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | in Factor and performs advanced optimizations. See | 
					
						
							|  |  |  | basis/compiler/compiler.factor. | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-05-17 04:11:48 -04:00
										 |  |  | The non-optimizing compiler compiles a quotation at a time by | 
					
						
							|  |  |  | concatenating machine code chunks; prolog, epilog, call word, jump to | 
					
						
							|  |  |  | word, etc. These machine code chunks are generated from Factor code in | 
					
						
							|  |  |  | basis/bootstrap/assembler/. | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | Calls to words and constant quotations (referenced by conditionals and dips) | 
					
						
							|  |  |  | are direct jumps to machine code blocks. Literals are also referenced directly | 
					
						
							|  |  |  | without going through the literal table. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | It actually does do a little bit of very simple optimization: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 1) Tail call optimization. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 2) If a quotation is determined to not call any other words (except for a few | 
					
						
							|  |  |  | special words which are open-coded, see below), then no prolog/epilog is | 
					
						
							|  |  |  | generated. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 3) When in tail position and immediately preceded by literal arguments, the | 
					
						
							|  |  |  | 'if' is generated inline, instead of as a call to the 'if' word. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 4) When preceded by a quotation, calls to 'dip', '2dip' and '3dip' are | 
					
						
							|  |  |  | open-coded as retain stack manipulation surrounding a subroutine call. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 5) Sub-primitives are primitive words which are implemented in assembly and not | 
					
						
							|  |  |  | in the VM. They are open-coded and no subroutine call is generated. This | 
					
						
							|  |  |  | includes stack shufflers, some fixnum arithmetic words, and words such as tag, | 
					
						
							|  |  |  | slot and eq?. A primitive call is relatively expensive (two subroutine calls) | 
					
						
							|  |  |  | so this results in a big speedup for relatively little effort. */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | void quotation_jit::init_quotation(cell quot) { | 
					
						
							|  |  |  |   elements = untag<quotation>(quot)->array; | 
					
						
							| 
									
										
										
										
											2009-11-24 23:38:15 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::primitive_call_p(cell i, cell length) { | 
					
						
							|  |  |  |   return (i + 2) <= length && array_nth(elements.untagged(), i + 1) == | 
					
						
							| 
									
										
										
										
											2015-07-31 14:31:47 -04:00
										 |  |  |       parent->special_objects[JIT_PRIMITIVE_WORD]; | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::fast_if_p(cell i, cell length) { | 
					
						
							|  |  |  |   return (i + 3) == length && | 
					
						
							|  |  |  |          tagged<object>(array_nth(elements.untagged(), i + 1)) | 
					
						
							|  |  |  |              .type_p(QUOTATION_TYPE) && | 
					
						
							|  |  |  |          array_nth(elements.untagged(), i + 2) == | 
					
						
							|  |  |  |              parent->special_objects[JIT_IF_WORD]; | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::fast_dip_p(cell i, cell length) { | 
					
						
							|  |  |  |   return (i + 2) <= length && array_nth(elements.untagged(), i + 1) == | 
					
						
							|  |  |  |                                   parent->special_objects[JIT_DIP_WORD]; | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::fast_2dip_p(cell i, cell length) { | 
					
						
							|  |  |  |   return (i + 2) <= length && array_nth(elements.untagged(), i + 1) == | 
					
						
							|  |  |  |                                   parent->special_objects[JIT_2DIP_WORD]; | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::fast_3dip_p(cell i, cell length) { | 
					
						
							|  |  |  |   return (i + 2) <= length && array_nth(elements.untagged(), i + 1) == | 
					
						
							|  |  |  |                                   parent->special_objects[JIT_3DIP_WORD]; | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::mega_lookup_p(cell i, cell length) { | 
					
						
							|  |  |  |   return (i + 4) <= length && | 
					
						
							|  |  |  |          tagged<object>(array_nth(elements.untagged(), i + 1)) | 
					
						
							|  |  |  |              .type_p(FIXNUM_TYPE) && | 
					
						
							|  |  |  |          tagged<object>(array_nth(elements.untagged(), i + 2)) | 
					
						
							|  |  |  |              .type_p(ARRAY_TYPE) && | 
					
						
							|  |  |  |          array_nth(elements.untagged(), i + 3) == | 
					
						
							|  |  |  |              parent->special_objects[MEGA_LOOKUP_WORD]; | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::declare_p(cell i, cell length) { | 
					
						
							|  |  |  |   return (i + 2) <= length && array_nth(elements.untagged(), i + 1) == | 
					
						
							|  |  |  |                                   parent->special_objects[JIT_DECLARE_WORD]; | 
					
						
							| 
									
										
										
										
											2009-09-25 19:08:21 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::special_subprimitive_p(cell obj) { | 
					
						
							|  |  |  |   // Subprimitives should be flagged with whether they require a stack frame.
 | 
					
						
							|  |  |  |   // See #295.
 | 
					
						
							|  |  |  |   return obj == parent->special_objects[SIGNAL_HANDLER_WORD] || | 
					
						
							|  |  |  |          obj == parent->special_objects[LEAF_SIGNAL_HANDLER_WORD] || | 
					
						
							|  |  |  |          obj == parent->special_objects[UNWIND_NATIVE_FRAMES_WORD]; | 
					
						
							| 
									
										
										
										
											2011-11-16 15:39:25 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::word_safepoint_p(cell obj) { | 
					
						
							|  |  |  |   return !special_subprimitive_p(obj); | 
					
						
							| 
									
										
										
										
											2011-11-16 15:39:25 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-07 22:45:40 -04:00
										 |  |  | /* true if there are no non-safepoint words in the quoation... */ | 
					
						
							|  |  |  | bool quotation_jit::no_non_safepoint_words_p() { | 
					
						
							| 
									
										
										
										
											2014-07-08 01:34:36 -04:00
										 |  |  |   cell length = array_capacity(elements.untagged()); | 
					
						
							|  |  |  |   for (cell i = 0; i < length; i++) { | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |     cell obj = array_nth(elements.untagged(), i); | 
					
						
							|  |  |  |     if (tagged<object>(obj).type() == WORD_TYPE && !word_safepoint_p(obj)) | 
					
						
							|  |  |  |       return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2011-11-16 15:39:25 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::trivial_quotation_p(array* elements) { | 
					
						
							|  |  |  |   return array_capacity(elements) == 1 && | 
					
						
							|  |  |  |          tagged<object>(array_nth(elements, 0)).type_p(WORD_TYPE); | 
					
						
							| 
									
										
										
										
											2009-10-06 02:31:39 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2014-02-13 23:42:44 -05:00
										 |  |  | /* Allocates memory (emit) */ | 
					
						
							| 
									
										
										
										
											2015-08-07 22:51:55 -04:00
										 |  |  | void quotation_jit::emit_epilog(bool needed) { | 
					
						
							|  |  |  |   if (needed) { | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |     emit(parent->special_objects[JIT_SAFEPOINT]); | 
					
						
							|  |  |  |     emit(parent->special_objects[JIT_EPILOG]); | 
					
						
							| 
									
										
										
										
											2015-08-07 22:51:55 -04:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2011-11-03 14:57:23 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-25 17:05:05 -04:00
										 |  |  | /* Allocates memory conditionally */ | 
					
						
							| 
									
										
										
										
											2015-07-15 14:21:20 -04:00
										 |  |  | void quotation_jit::emit_quotation(cell quot_) { | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   data_root<quotation> quot(quot_, parent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   array* elements = untag<array>(quot->array); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* If the quotation consists of a single word, compile a direct call
 | 
					
						
							|  |  |  |      to the word. */ | 
					
						
							|  |  |  |   if (trivial_quotation_p(elements)) | 
					
						
							|  |  |  |     literal(array_nth(elements, 0)); | 
					
						
							|  |  |  |   else { | 
					
						
							|  |  |  |     if (compiling) | 
					
						
							| 
									
										
										
										
											2015-07-15 14:13:52 -04:00
										 |  |  |       parent->jit_compile_quotation(quot.value(), relocate); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |     literal(quot.value()); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-10-06 02:31:39 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-07 22:51:55 -04:00
										 |  |  | /* Allocates memory (parameter(), literal(), emit_epilog, emit_with_literal)*/ | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | void quotation_jit::iterate_quotation() { | 
					
						
							| 
									
										
										
										
											2015-08-07 22:45:40 -04:00
										 |  |  |   bool no_non_safepoint_words = no_non_safepoint_words_p(); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   set_position(0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-07 22:51:55 -04:00
										 |  |  |   if (no_non_safepoint_words) { | 
					
						
							|  |  |  |     emit(parent->special_objects[JIT_SAFEPOINT]); | 
					
						
							|  |  |  |     emit(parent->special_objects[JIT_PROLOG]); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   cell length = array_capacity(elements.untagged()); | 
					
						
							|  |  |  |   bool tail_call = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-14 16:31:04 -04:00
										 |  |  |   for (cell i = 0; i < length; i++) { | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |     set_position(i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     data_root<object> obj(array_nth(elements.untagged(), i), parent); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (obj.type()) { | 
					
						
							|  |  |  |       case WORD_TYPE: | 
					
						
							|  |  |  |         /* Sub-primitives */ | 
					
						
							|  |  |  |         if (to_boolean(obj.as<word>()->subprimitive)) { | 
					
						
							|  |  |  |           tail_call = emit_subprimitive(obj.value(),     /* word */ | 
					
						
							|  |  |  |                                         i == length - 1, /* tail_call_p */ | 
					
						
							| 
									
										
										
										
											2015-08-07 22:45:40 -04:00
										 |  |  |                                         no_non_safepoint_words);    /* stack_frame_p */ | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |         }                                                /* Everything else */ | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  |         else if (i == length - 1) { | 
					
						
							| 
									
										
										
										
											2015-08-07 22:51:55 -04:00
										 |  |  |           emit_epilog(no_non_safepoint_words); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           tail_call = true; | 
					
						
							|  |  |  |           word_jump(obj.value()); | 
					
						
							|  |  |  |         } else | 
					
						
							|  |  |  |           word_call(obj.value()); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case WRAPPER_TYPE: | 
					
						
							|  |  |  |         push(obj.as<wrapper>()->object); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case BYTE_ARRAY_TYPE: | 
					
						
							|  |  |  |         /* Primitive calls */ | 
					
						
							|  |  |  |         if (primitive_call_p(i, length)) { | 
					
						
							|  |  |  | /* On x86-64 and PowerPC, the VM pointer is stored in
 | 
					
						
							|  |  |  |    a register; on other platforms, the RT_VM relocation | 
					
						
							|  |  |  |    is used and it needs an offset parameter */ | 
					
						
							| 
									
										
										
										
											2010-01-10 07:20:32 -05:00
										 |  |  | #ifdef FACTOR_X86
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           parameter(tag_fixnum(0)); | 
					
						
							| 
									
										
										
										
											2009-12-28 09:08:15 -05:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           parameter(obj.value()); | 
					
						
							|  |  |  |           parameter(false_object); | 
					
						
							| 
									
										
										
										
											2011-05-20 18:11:50 -04:00
										 |  |  | #ifdef FACTOR_PPC_TOC
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           parameter(obj.value()); | 
					
						
							|  |  |  |           parameter(false_object); | 
					
						
							| 
									
										
										
										
											2011-05-20 18:11:50 -04:00
										 |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           emit(parent->special_objects[JIT_PRIMITIVE]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           i++; | 
					
						
							|  |  |  |         } else | 
					
						
							|  |  |  |           push(obj.value()); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case QUOTATION_TYPE: | 
					
						
							|  |  |  |         /* 'if' preceded by two literal quotations (this is why if and ? are
 | 
					
						
							|  |  |  |            mutually recursive in the library, but both still work) */ | 
					
						
							|  |  |  |         if (fast_if_p(i, length)) { | 
					
						
							| 
									
										
										
										
											2015-08-07 22:51:55 -04:00
										 |  |  |           emit_epilog(no_non_safepoint_words); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           tail_call = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-15 14:21:20 -04:00
										 |  |  |           emit_quotation(array_nth(elements.untagged(), i)); | 
					
						
							|  |  |  |           emit_quotation(array_nth(elements.untagged(), i + 1)); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           emit(parent->special_objects[JIT_IF]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           i += 2; | 
					
						
							|  |  |  |         } /* dip */ | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  |         else if (fast_dip_p(i, length)) { | 
					
						
							| 
									
										
										
										
											2015-07-15 14:21:20 -04:00
										 |  |  |           emit_quotation(obj.value()); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           emit(parent->special_objects[JIT_DIP]); | 
					
						
							|  |  |  |           i++; | 
					
						
							|  |  |  |         } /* 2dip */ | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  |         else if (fast_2dip_p(i, length)) { | 
					
						
							| 
									
										
										
										
											2015-07-15 14:21:20 -04:00
										 |  |  |           emit_quotation(obj.value()); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           emit(parent->special_objects[JIT_2DIP]); | 
					
						
							|  |  |  |           i++; | 
					
						
							|  |  |  |         } /* 3dip */ | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  |         else if (fast_3dip_p(i, length)) { | 
					
						
							| 
									
										
										
										
											2015-07-15 14:21:20 -04:00
										 |  |  |           emit_quotation(obj.value()); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           emit(parent->special_objects[JIT_3DIP]); | 
					
						
							|  |  |  |           i++; | 
					
						
							|  |  |  |         } else | 
					
						
							|  |  |  |           push(obj.value()); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       case ARRAY_TYPE: | 
					
						
							|  |  |  |         /* Method dispatch */ | 
					
						
							|  |  |  |         if (mega_lookup_p(i, length)) { | 
					
						
							|  |  |  |           fixnum index = untag_fixnum(array_nth(elements.untagged(), i + 1)); | 
					
						
							|  |  |  |           /* Load the object from the datastack, then remove our stack frame. */ | 
					
						
							|  |  |  |           emit_with_literal(parent->special_objects[PIC_LOAD], | 
					
						
							|  |  |  |                             tag_fixnum(-index * sizeof(cell))); | 
					
						
							| 
									
										
										
										
											2015-08-07 22:51:55 -04:00
										 |  |  |           emit_epilog(no_non_safepoint_words); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           tail_call = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           emit_mega_cache_lookup(array_nth(elements.untagged(), i), index, | 
					
						
							|  |  |  |                                  array_nth(elements.untagged(), i + 2)); | 
					
						
							|  |  |  |           i += 3; | 
					
						
							|  |  |  |         } /* Non-optimizing compiler ignores declarations */ | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  |         else if (declare_p(i, length)) | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           i++; | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |           push(obj.value()); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |       default: | 
					
						
							|  |  |  |         push(obj.value()); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!tail_call) { | 
					
						
							|  |  |  |     set_position(length); | 
					
						
							| 
									
										
										
										
											2015-08-07 22:51:55 -04:00
										 |  |  |     emit_epilog(no_non_safepoint_words); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |     emit(parent->special_objects[JIT_RETURN]); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | cell quotation_jit::word_stack_frame_size(cell obj) { | 
					
						
							|  |  |  |   if (special_subprimitive_p(obj)) | 
					
						
							|  |  |  |     return SIGNAL_HANDLER_STACK_FRAME_SIZE; | 
					
						
							| 
									
										
										
										
											2015-08-30 06:15:22 -04:00
										 |  |  |   return JIT_FRAME_SIZE; | 
					
						
							| 
									
										
										
										
											2011-12-13 20:04:35 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2015-07-15 14:13:52 -04:00
										 |  |  | code_block* factor_vm::jit_compile_quotation(cell owner_, cell quot_, | 
					
						
							|  |  |  |                                              bool relocating) { | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   data_root<object> owner(owner_, this); | 
					
						
							|  |  |  |   data_root<quotation> quot(quot_, this); | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   quotation_jit compiler(owner.value(), true, relocating, this); | 
					
						
							|  |  |  |   compiler.init_quotation(quot.value()); | 
					
						
							|  |  |  |   compiler.iterate_quotation(); | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   cell frame_size = compiler.word_stack_frame_size(owner_); | 
					
						
							| 
									
										
										
										
											2011-11-30 15:39:21 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   code_block* compiled = compiler.to_code_block(frame_size); | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   if (relocating) | 
					
						
							|  |  |  |     initialize_code_block(compiled); | 
					
						
							| 
									
										
										
										
											2009-11-24 23:38:15 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   return compiled; | 
					
						
							| 
									
										
										
										
											2009-11-24 23:38:15 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-25 17:05:05 -04:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2015-07-15 14:13:52 -04:00
										 |  |  | void factor_vm::jit_compile_quotation(cell quot_, bool relocating) { | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   data_root<quotation> quot(quot_, this); | 
					
						
							| 
									
										
										
										
											2015-07-15 14:13:52 -04:00
										 |  |  |   if (!quotation_compiled_p(quot.untagged())) { | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |     code_block* compiled = | 
					
						
							| 
									
										
										
										
											2015-07-15 14:13:52 -04:00
										 |  |  |         jit_compile_quotation(quot.value(), quot.value(), relocating); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |     quot.untagged()->entry_point = compiled->entry_point(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-25 17:05:05 -04:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2015-07-15 14:13:52 -04:00
										 |  |  | void factor_vm::primitive_jit_compile() { | 
					
						
							|  |  |  |   jit_compile_quotation(ctx->pop(), true); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 05:51:19 -05:00
										 |  |  | cell factor_vm::lazy_jit_compile_entry_point() { | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   return untag<word>(special_objects[LAZY_JIT_COMPILE_WORD])->entry_point; | 
					
						
							| 
									
										
										
										
											2010-01-05 21:47:36 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | /* push a new quotation on the stack */ | 
					
						
							| 
									
										
										
										
											2013-03-25 17:05:05 -04:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | void factor_vm::primitive_array_to_quotation() { | 
					
						
							|  |  |  |   quotation* quot = allot<quotation>(sizeof(quotation)); | 
					
						
							| 
									
										
										
										
											2010-01-05 21:47:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   quot->array = ctx->peek(); | 
					
						
							|  |  |  |   quot->cached_effect = false_object; | 
					
						
							|  |  |  |   quot->cache_counter = false_object; | 
					
						
							|  |  |  |   quot->entry_point = lazy_jit_compile_entry_point(); | 
					
						
							| 
									
										
										
										
											2010-01-05 21:47:36 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   ctx->replace(tag<quotation>(quot)); | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2012-08-16 00:30:55 -04:00
										 |  |  | /* Allocates memory (from_unsigned_cell) */ | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | void factor_vm::primitive_quotation_code() { | 
					
						
							|  |  |  |   data_root<quotation> quot(ctx->pop(), this); | 
					
						
							| 
									
										
										
										
											2010-01-18 02:51:27 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-01-07 05:51:19 -05:00
										 |  |  |   ctx->push(from_unsigned_cell(quot->entry_point)); | 
					
						
							| 
									
										
										
										
											2013-05-13 00:53:47 -04:00
										 |  |  |   ctx->push(from_unsigned_cell((cell)quot->code() + quot->code()->size())); | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-02 10:19:09 -04:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | fixnum factor_vm::quot_code_offset_to_scan(cell quot_, cell offset) { | 
					
						
							|  |  |  |   data_root<quotation> quot(quot_, this); | 
					
						
							|  |  |  |   data_root<array> array(quot->array, this); | 
					
						
							| 
									
										
										
										
											2009-05-02 10:19:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   quotation_jit compiler(quot.value(), false, false, this); | 
					
						
							|  |  |  |   compiler.init_quotation(quot.value()); | 
					
						
							|  |  |  |   compiler.compute_position(offset); | 
					
						
							|  |  |  |   compiler.iterate_quotation(); | 
					
						
							| 
									
										
										
										
											2009-05-02 10:19:09 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   return compiler.get_position(); | 
					
						
							| 
									
										
										
										
											2009-05-02 10:19:09 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-25 17:05:05 -04:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | cell factor_vm::lazy_jit_compile(cell quot_) { | 
					
						
							|  |  |  |   data_root<quotation> quot(quot_, this); | 
					
						
							| 
									
										
										
										
											2010-01-19 08:48:31 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-15 14:13:52 -04:00
										 |  |  |   FACTOR_ASSERT(!quotation_compiled_p(quot.untagged())); | 
					
						
							| 
									
										
										
										
											2010-01-19 08:48:31 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-15 14:13:52 -04:00
										 |  |  |   code_block* compiled = | 
					
						
							|  |  |  |       jit_compile_quotation(quot.value(), quot.value(), true); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   quot.untagged()->entry_point = compiled->entry_point(); | 
					
						
							| 
									
										
										
										
											2010-01-19 08:48:31 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   return quot.value(); | 
					
						
							| 
									
										
										
										
											2009-05-04 02:00:30 -04:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2009-05-04 02:46:13 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-25 17:05:05 -04:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | VM_C_API cell lazy_jit_compile(cell quot, factor_vm* parent) { | 
					
						
							|  |  |  |   return parent->lazy_jit_compile(quot); | 
					
						
							| 
									
										
										
										
											2009-08-17 16:37:08 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-15 14:13:52 -04:00
										 |  |  | bool factor_vm::quotation_compiled_p(quotation* quot) { | 
					
						
							| 
									
										
										
										
											2015-01-07 05:51:19 -05:00
										 |  |  |   return quot->entry_point != 0 && | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |          quot->entry_point != lazy_jit_compile_entry_point(); | 
					
						
							| 
									
										
										
										
											2010-01-06 05:49:14 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-15 14:13:52 -04:00
										 |  |  | void factor_vm::primitive_quotation_compiled_p() { | 
					
						
							|  |  |  |   quotation* quot = untag_check<quotation>(ctx->pop()); | 
					
						
							|  |  |  |   ctx->push(tag_boolean(quotation_compiled_p(quot))); | 
					
						
							| 
									
										
										
										
											2010-01-05 21:47:36 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-03-25 17:05:05 -04:00
										 |  |  | /* Allocates memory */ | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | void factor_vm::initialize_all_quotations() { | 
					
						
							| 
									
										
										
										
											2015-07-31 14:31:47 -04:00
										 |  |  |   cell all_quots = instances(QUOTATION_TYPE); | 
					
						
							|  |  |  |   data_root<array> quotations(all_quots, this); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   cell length = array_capacity(quotations.untagged()); | 
					
						
							|  |  |  |   for (cell i = 0; i < length; i++) { | 
					
						
							|  |  |  |     data_root<quotation> quot(array_nth(quotations.untagged(), i), this); | 
					
						
							|  |  |  |     if (!quot->entry_point) | 
					
						
							|  |  |  |       quot.untagged()->entry_point = lazy_jit_compile_entry_point(); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2009-05-12 04:09:15 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-04 02:46:13 -04:00
										 |  |  | } |