| 
									
										
										
										
											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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // Simple non-optimizing compiler.
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // This is one of the two compilers implementing Factor; the second one is
 | 
					
						
							|  |  |  | // written in Factor and performs advanced optimizations. See
 | 
					
						
							|  |  |  | // basis/compiler/compiler.factor.
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -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.
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // It actually does do a little bit of very simple optimization:
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // 1) Tail call optimization.
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // 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.
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // 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.
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // 4) When preceded by a quotation, calls to 'dip', '2dip' and '3dip' are
 | 
					
						
							|  |  |  | // open-coded as retain stack manipulation surrounding a subroutine call.
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // 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.
 | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-30 13:32:03 -04:00
										 |  |  | inline cell quotation_jit::nth(cell index) { | 
					
						
							|  |  |  |   return array_nth(elements.untagged(), index); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | void quotation_jit::init_quotation(cell quot) { | 
					
						
							| 
									
										
										
										
											2016-11-22 21:25:07 -05:00
										 |  |  |   elements.set_value(untag<quotation>(quot)->array); | 
					
						
							| 
									
										
										
										
											2009-11-24 23:38:15 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::fast_if_p(cell i, cell length) { | 
					
						
							|  |  |  |   return (i + 3) == length && | 
					
						
							| 
									
										
										
										
											2016-11-11 18:19:52 -05:00
										 |  |  |       TAG(nth(i + 1)) == QUOTATION_TYPE && | 
					
						
							|  |  |  |       nth(i + 2) == parent->special_objects[JIT_IF_WORD]; | 
					
						
							| 
									
										
										
										
											2015-11-22 17:07:18 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool quotation_jit::primitive_call_p(cell i, cell length) { | 
					
						
							| 
									
										
										
										
											2016-11-11 18:19:52 -05:00
										 |  |  |   cell jit_primitive_word = parent->special_objects[JIT_PRIMITIVE_WORD]; | 
					
						
							|  |  |  |   return (i + 2) <= length && nth(i + 1) == jit_primitive_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) { | 
					
						
							| 
									
										
										
										
											2016-11-11 18:19:52 -05:00
										 |  |  |   cell jit_dip_word = parent->special_objects[JIT_DIP_WORD]; | 
					
						
							|  |  |  |   return (i + 2) <= length && nth(i + 1) == 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) { | 
					
						
							| 
									
										
										
										
											2016-11-11 18:19:52 -05:00
										 |  |  |   cell jit_2dip_word = parent->special_objects[JIT_2DIP_WORD]; | 
					
						
							|  |  |  |   return (i + 2) <= length && nth(i + 1) == 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) { | 
					
						
							| 
									
										
										
										
											2016-11-11 18:19:52 -05:00
										 |  |  |   cell jit_3dip_word = parent->special_objects[JIT_3DIP_WORD]; | 
					
						
							|  |  |  |   return (i + 2) <= length && nth(i + 1) == jit_3dip_word; | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-22 17:07:18 -05:00
										 |  |  | bool quotation_jit::declare_p(cell i, cell length) { | 
					
						
							| 
									
										
										
										
											2016-11-11 18:19:52 -05:00
										 |  |  |   cell jit_declare_word = parent->special_objects[JIT_DECLARE_WORD]; | 
					
						
							|  |  |  |   return (i + 2) <= length && nth(i + 1) == jit_declare_word; | 
					
						
							| 
									
										
										
										
											2009-05-02 05:04:19 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-22 17:07:18 -05:00
										 |  |  | bool quotation_jit::mega_lookup_p(cell i, cell length) { | 
					
						
							|  |  |  |   return (i + 4) <= length && | 
					
						
							| 
									
										
										
										
											2016-11-11 18:19:52 -05:00
										 |  |  |       TAG(nth(i + 1)) == FIXNUM_TYPE && | 
					
						
							|  |  |  |       TAG(nth(i + 2)) == ARRAY_TYPE && | 
					
						
							|  |  |  |       nth(i + 3) == parent->special_objects[MEGA_LOOKUP_WORD]; | 
					
						
							| 
									
										
										
										
											2009-09-25 19:08:21 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // Subprimitives should be flagged with whether they require a stack frame.
 | 
					
						
							|  |  |  | // See #295.
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | bool quotation_jit::special_subprimitive_p(cell obj) { | 
					
						
							|  |  |  |   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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // All quotations wants a stack frame, except if they contain:
 | 
					
						
							|  |  |  | //   1) calls to the special subprimitives, see #295.
 | 
					
						
							|  |  |  | //   2) mega cache lookups, see #651
 | 
					
						
							| 
									
										
										
										
											2016-05-29 22:57:14 -04:00
										 |  |  | bool quotation_jit::stack_frame_p() { | 
					
						
							| 
									
										
										
										
											2014-07-08 01:34:36 -04:00
										 |  |  |   cell length = array_capacity(elements.untagged()); | 
					
						
							|  |  |  |   for (cell i = 0; i < length; i++) { | 
					
						
							| 
									
										
										
										
											2016-11-11 18:19:52 -05:00
										 |  |  |     cell obj = nth(i); | 
					
						
							|  |  |  |     cell tag = TAG(obj); | 
					
						
							| 
									
										
										
										
											2016-05-30 02:23:18 -04:00
										 |  |  |     if ((tag == WORD_TYPE && special_subprimitive_p(obj)) || | 
					
						
							|  |  |  |         (tag == ARRAY_TYPE && mega_lookup_p(i, length))) | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |       return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return true; | 
					
						
							| 
									
										
										
										
											2011-11-16 15:39:25 -05:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-29 23:19:32 -04:00
										 |  |  | static bool trivial_quotation_p(array* elements) { | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   return array_capacity(elements) == 1 && | 
					
						
							| 
									
										
										
										
											2015-11-22 17:07:18 -05:00
										 |  |  |       TAG(array_nth(elements, 0)) == WORD_TYPE; | 
					
						
							| 
									
										
										
										
											2009-10-06 02:31:39 -04:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04: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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -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); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |   // If the quotation consists of a single word, compile a direct call
 | 
					
						
							|  |  |  |   // to the word.
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |   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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // Allocates memory (parameter(), literal(), emit_epilog, emit_with_literal)
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | void quotation_jit::iterate_quotation() { | 
					
						
							| 
									
										
										
										
											2016-05-29 22:57:14 -04:00
										 |  |  |   bool stack_frame = stack_frame_p(); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   set_position(0); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-05-29 22:57:14 -04:00
										 |  |  |   if (stack_frame) { | 
					
						
							| 
									
										
										
										
											2015-08-07 22:51:55 -04:00
										 |  |  |     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); | 
					
						
							| 
									
										
										
										
											2016-05-30 13:32:03 -04:00
										 |  |  |     data_root<object> obj(nth(i), parent); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     switch (obj.type()) { | 
					
						
							|  |  |  |       case WORD_TYPE: | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |         // Sub-primitives
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |         if (to_boolean(obj.as<word>()->subprimitive)) { | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |           tail_call = emit_subprimitive(obj.value(),     // word
 | 
					
						
							|  |  |  |                                         i == length - 1, // tail_call_p
 | 
					
						
							|  |  |  |                                         stack_frame);    // stack_frame_p
 | 
					
						
							|  |  |  |         }                                                // Everything else
 | 
					
						
							| 
									
										
										
										
											2014-11-25 04:12:45 -05:00
										 |  |  |         else if (i == length - 1) { | 
					
						
							| 
									
										
										
										
											2016-05-29 22:57:14 -04:00
										 |  |  |           emit_epilog(stack_frame); | 
					
						
							| 
									
										
										
										
											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: | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |         // Primitive calls
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |         if (primitive_call_p(i, length)) { | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // 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: | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |         // 'if' preceded by two literal quotations (this is why if and ? are
 | 
					
						
							|  |  |  |         // mutually recursive in the library, but both still work)
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |         if (fast_if_p(i, length)) { | 
					
						
							| 
									
										
										
										
											2016-05-29 22:57:14 -04:00
										 |  |  |           emit_epilog(stack_frame); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           tail_call = true; | 
					
						
							| 
									
										
										
										
											2016-05-30 13:32:03 -04:00
										 |  |  |           emit_quotation(nth(i)); | 
					
						
							|  |  |  |           emit_quotation(nth(i + 1)); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           emit(parent->special_objects[JIT_IF]); | 
					
						
							|  |  |  |           i += 2; | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |         } // 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++; | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |         } // 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++; | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |         } // 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: | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |         // Method dispatch
 | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |         if (mega_lookup_p(i, length)) { | 
					
						
							|  |  |  |           tail_call = true; | 
					
						
							| 
									
										
										
										
											2016-05-30 13:32:03 -04:00
										 |  |  |           emit_mega_cache_lookup(nth(i), untag_fixnum(nth(i + 1)), nth(i + 2)); | 
					
						
							| 
									
										
										
										
											2013-05-11 22:27:06 -04:00
										 |  |  |           i += 3; | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |         } // 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); | 
					
						
							| 
									
										
										
										
											2016-05-29 22:57:14 -04:00
										 |  |  |     emit_epilog(stack_frame); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // Allocates memory
 | 
					
						
							| 
									
										
										
										
											2016-05-29 21:03:10 -04:00
										 |  |  | void quotation_jit::emit_mega_cache_lookup(cell methods_, fixnum index, | 
					
						
							|  |  |  |                                            cell cache_) { | 
					
						
							|  |  |  |   data_root<array> methods(methods_, parent); | 
					
						
							|  |  |  |   data_root<array> cache(cache_, parent); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |   // Load the object from the datastack.
 | 
					
						
							| 
									
										
										
										
											2016-05-30 02:23:18 -04:00
										 |  |  |   emit_with_literal(parent->special_objects[PIC_LOAD], | 
					
						
							|  |  |  |                     tag_fixnum(-index * sizeof(cell))); | 
					
						
							| 
									
										
										
										
											2016-05-29 21:03:10 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |   // Do a cache lookup.
 | 
					
						
							| 
									
										
										
										
											2016-05-29 21:03:10 -04:00
										 |  |  |   emit_with_literal(parent->special_objects[MEGA_LOOKUP], cache.value()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |   // If we end up here, the cache missed.
 | 
					
						
							| 
									
										
										
										
											2016-05-29 21:03:10 -04:00
										 |  |  |   emit(parent->special_objects[JIT_PROLOG]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |   // Push index, method table and cache on the stack.
 | 
					
						
							| 
									
										
										
										
											2016-05-29 21:03:10 -04:00
										 |  |  |   push(methods.value()); | 
					
						
							|  |  |  |   push(tag_fixnum(index)); | 
					
						
							|  |  |  |   push(cache.value()); | 
					
						
							|  |  |  |   word_call(parent->special_objects[MEGA_MISS_WORD]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  |   // Now the new method has been stored into the cache, and its on
 | 
					
						
							|  |  |  |   // the stack.
 | 
					
						
							| 
									
										
										
										
											2016-05-29 21:03:10 -04:00
										 |  |  |   emit(parent->special_objects[JIT_EPILOG]); | 
					
						
							|  |  |  |   emit(parent->special_objects[JIT_EXECUTE]); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-10-19 03:28:59 -04:00
										 |  |  |   code_block* compiled = compiler.to_code_block(CODE_BLOCK_UNOPTIMIZED, | 
					
						
							|  |  |  |                                                 frame_size); | 
					
						
							| 
									
										
										
										
											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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -04:00
										 |  |  | // push a new quotation on the stack
 | 
					
						
							|  |  |  | // 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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-08-21 10:26:04 -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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-04 02:46:13 -04:00
										 |  |  | } |