| 
									
										
										
										
											2010-06-13 17:36:08 -04:00
										 |  |  | ! Copyright (C) 2009, 2010 Slava Pestov. | 
					
						
							| 
									
										
										
										
											2009-07-22 07:07:28 -04:00
										 |  |  | ! See http://factorcode.org/license.txt for BSD license. | 
					
						
							| 
									
										
										
										
											2015-04-21 16:45:38 -04:00
										 |  |  | USING: accessors assocs combinators combinators.short-circuit | 
					
						
							| 
									
										
										
										
											2017-02-07 21:54:24 -05:00
										 |  |  | compiler.cfg.def-use compiler.cfg.instructions | 
					
						
							|  |  |  | compiler.cfg.predecessors compiler.cfg.registers | 
					
						
							|  |  |  | compiler.cfg.rpo compiler.cfg.ssa.destruction.leaders | 
					
						
							|  |  |  | compiler.cfg.utilities compiler.utilities cpu.architecture | 
					
						
							|  |  |  | deques dlists fry kernel locals namespaces sequences sets ;
 | 
					
						
							| 
									
										
										
										
											2009-07-22 07:07:28 -04:00
										 |  |  | IN: compiler.cfg.liveness | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-25 17:36:58 -04:00
										 |  |  | SYMBOL: live-ins | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : live-in ( bb -- set )
 | 
					
						
							|  |  |  |     live-ins get at ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SYMBOL: live-outs | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : live-out ( bb -- set )
 | 
					
						
							|  |  |  |     live-outs get at ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | SYMBOL: edge-live-ins | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : edge-live-in ( predecessor basic-block -- set )
 | 
					
						
							|  |  |  |     edge-live-ins get at at ;
 | 
					
						
							| 
									
										
										
										
											2009-07-22 07:07:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | SYMBOL: base-pointers | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 02:56:23 -04:00
										 |  |  | GENERIC: visit-insn ( live-set insn -- )
 | 
					
						
							| 
									
										
										
										
											2009-07-27 01:31:21 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 02:56:23 -04:00
										 |  |  | ! This would be much better if live-set was a real set | 
					
						
							|  |  |  | : kill-defs ( live-set insn -- )
 | 
					
						
							| 
									
										
										
										
											2015-04-16 01:03:50 -04:00
										 |  |  |     defs-vregs [ ?leader ] map
 | 
					
						
							| 
									
										
										
										
											2015-05-12 22:08:42 -04:00
										 |  |  |     '[ drop ?leader _ in? ] assoc-reject! drop ; inline
 | 
					
						
							| 
									
										
										
										
											2009-07-28 07:45:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 02:56:23 -04:00
										 |  |  | : gen-uses ( live-set insn -- )
 | 
					
						
							|  |  |  |     uses-vregs [ swap conjoin ] with each ; inline
 | 
					
						
							| 
									
										
										
										
											2010-06-13 17:36:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 02:56:23 -04:00
										 |  |  | M: vreg-insn visit-insn ( live-set insn -- )
 | 
					
						
							|  |  |  |     [ kill-defs ] [ gen-uses ] 2bi ;
 | 
					
						
							| 
									
										
										
										
											2010-06-13 17:36:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | DEFER: lookup-base-pointer | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-01 17:16:51 -05:00
										 |  |  | GENERIC: lookup-base-pointer* ( vreg insn -- vreg/f )
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-01 17:16:51 -05:00
										 |  |  | M: ##tagged>integer lookup-base-pointer* nip src>> ;
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | M: ##unbox-any-c-ptr lookup-base-pointer* | 
					
						
							|  |  |  |     ! If the input to unbox-any-c-ptr was an alien and not a | 
					
						
							|  |  |  |     ! byte array, then the derived pointer will be outside of | 
					
						
							|  |  |  |     ! the data heap. The GC has to handle this case and ignore | 
					
						
							|  |  |  |     ! it. | 
					
						
							| 
									
										
										
										
											2013-01-01 17:16:51 -05:00
										 |  |  |     nip src>> ;
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-01 17:16:51 -05:00
										 |  |  | M: ##copy lookup-base-pointer* nip src>> lookup-base-pointer ;
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-01 17:16:51 -05:00
										 |  |  | M: ##add-imm lookup-base-pointer* nip src1>> lookup-base-pointer ;
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-01 17:16:51 -05:00
										 |  |  | M: ##sub-imm lookup-base-pointer* nip src1>> lookup-base-pointer ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: ##parallel-copy lookup-base-pointer* values>> value-at ;
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | M: ##add lookup-base-pointer* | 
					
						
							|  |  |  |     ! If both operands have a base pointer, then the user better | 
					
						
							|  |  |  |     ! not be doing memory reads and writes on the object, since | 
					
						
							|  |  |  |     ! we don't give it a base pointer in that case at all. | 
					
						
							| 
									
										
										
										
											2013-01-01 17:16:51 -05:00
										 |  |  |     nip [ src1>> ] [ src2>> ] bi [ lookup-base-pointer ] bi@ xor ;
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | M: ##sub lookup-base-pointer* | 
					
						
							| 
									
										
										
										
											2013-01-01 17:16:51 -05:00
										 |  |  |     nip src1>> lookup-base-pointer ;
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2013-01-01 17:16:51 -05:00
										 |  |  | M: vreg-insn lookup-base-pointer* 2drop f ;
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 01:03:50 -04:00
										 |  |  | : lookup-base-pointer ( vreg -- vreg/f )
 | 
					
						
							| 
									
										
										
										
											2013-01-01 17:47:54 -05:00
										 |  |  |     base-pointers get ?at [ | 
					
						
							|  |  |  |         f over base-pointers get set-at
 | 
					
						
							|  |  |  |         [ dup ?leader insn-of lookup-base-pointer* ] keep
 | 
					
						
							|  |  |  |         dupd base-pointers get set-at
 | 
					
						
							|  |  |  |     ] unless ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  | :: visit-derived-root ( vreg derived-roots gc-roots -- )
 | 
					
						
							|  |  |  |     vreg lookup-base-pointer :> base | 
					
						
							|  |  |  |     base [ | 
					
						
							|  |  |  |         { vreg base } derived-roots push
 | 
					
						
							|  |  |  |         base gc-roots adjoin | 
					
						
							|  |  |  |     ] when ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : visit-gc-root ( vreg derived-roots gc-roots -- )
 | 
					
						
							|  |  |  |     pick rep-of { | 
					
						
							|  |  |  |         { tagged-rep [ nip adjoin ] } | 
					
						
							|  |  |  |         { int-rep [ visit-derived-root ] } | 
					
						
							| 
									
										
										
										
											2012-10-23 15:21:30 -04:00
										 |  |  |         [ 4drop ] | 
					
						
							| 
									
										
										
										
											2010-09-27 01:20:50 -04:00
										 |  |  |     } case ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : gc-roots ( live-set -- derived-roots gc-roots )
 | 
					
						
							| 
									
										
										
										
											2015-04-21 16:45:38 -04:00
										 |  |  |     keys V{ } clone HS{ } clone
 | 
					
						
							|  |  |  |     [ '[ _ _ visit-gc-root ] each ] 2keep members ;
 | 
					
						
							| 
									
										
										
										
											2010-09-25 17:36:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 02:56:23 -04:00
										 |  |  | : fill-gc-map ( live-set gc-map -- )
 | 
					
						
							| 
									
										
										
										
											2015-04-21 16:45:38 -04:00
										 |  |  |     [ gc-roots ] dip [ gc-roots<< ] [ derived-roots<< ] bi ;
 | 
					
						
							| 
									
										
										
										
											2010-06-13 17:36:08 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 02:56:23 -04:00
										 |  |  | M: gc-map-insn visit-insn ( live-set insn -- )
 | 
					
						
							|  |  |  |     [ kill-defs ] [ gc-map>> fill-gc-map ] [ gen-uses ] 2tri ;
 | 
					
						
							| 
									
										
										
										
											2010-06-13 17:36:08 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | M: ##phi visit-insn kill-defs ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 02:56:23 -04:00
										 |  |  | M: insn visit-insn 2drop ;
 | 
					
						
							| 
									
										
										
										
											2009-07-28 07:45:02 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-16 02:56:23 -04:00
										 |  |  | : transfer-liveness ( live-set insns -- )
 | 
					
						
							|  |  |  |     <reversed> [ visit-insn ] with each ;
 | 
					
						
							| 
									
										
										
										
											2009-07-22 07:07:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-25 17:36:58 -04:00
										 |  |  | : compute-live-in ( basic-block -- live-in )
 | 
					
						
							| 
									
										
										
										
											2015-04-16 02:56:23 -04:00
										 |  |  |     [ live-out clone dup ] keep instructions>> transfer-liveness ;
 | 
					
						
							| 
									
										
										
										
											2010-09-25 17:36:58 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : compute-edge-live-in ( basic-block -- edge-live-in )
 | 
					
						
							|  |  |  |     H{ } clone [ | 
					
						
							|  |  |  |         '[ inputs>> [ swap _ conjoin-at ] assoc-each ] each-phi | 
					
						
							|  |  |  |     ] keep ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : update-live-in ( basic-block -- changed? )
 | 
					
						
							|  |  |  |     [ [ compute-live-in ] keep live-ins get maybe-set-at ] | 
					
						
							|  |  |  |     [ [ compute-edge-live-in ] keep edge-live-ins get maybe-set-at ] | 
					
						
							|  |  |  |     bi or ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : compute-live-out ( basic-block -- live-out )
 | 
					
						
							|  |  |  |     [ successors>> [ live-in ] map ] | 
					
						
							|  |  |  |     [ dup successors>> [ edge-live-in ] with map ] bi
 | 
					
						
							|  |  |  |     append assoc-combine ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : update-live-out ( basic-block -- changed? )
 | 
					
						
							|  |  |  |     [ compute-live-out ] keep
 | 
					
						
							|  |  |  |     live-outs get maybe-set-at ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-21 16:45:38 -04:00
										 |  |  | : update-live-out/in ( basic-block -- changed? )
 | 
					
						
							|  |  |  |     { [ update-live-out ] [ update-live-in ] } 1&& ;
 | 
					
						
							| 
									
										
										
										
											2010-09-25 17:36:58 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-21 16:45:38 -04:00
										 |  |  | : liveness-step ( basic-block -- basic-blocks )
 | 
					
						
							|  |  |  |     [ update-live-out/in ] keep predecessors>> { } ? ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : init-liveness ( -- )
 | 
					
						
							| 
									
										
										
										
											2016-03-29 20:14:42 -04:00
										 |  |  |     H{ } clone live-ins namespaces:set | 
					
						
							|  |  |  |     H{ } clone edge-live-ins namespaces:set | 
					
						
							|  |  |  |     H{ } clone live-outs namespaces:set | 
					
						
							|  |  |  |     H{ } clone base-pointers namespaces:set ;
 | 
					
						
							| 
									
										
										
										
											2014-12-10 12:24:12 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-04-21 16:45:38 -04:00
										 |  |  | : compute-live-sets ( cfg -- )
 | 
					
						
							|  |  |  |     init-liveness | 
					
						
							|  |  |  |     dup needs-predecessors dup compute-insns | 
					
						
							|  |  |  |     post-order <hashed-dlist> [ push-all-front ] keep
 | 
					
						
							|  |  |  |     [ liveness-step ] slurp/replenish-deque ;
 | 
					
						
							| 
									
										
										
										
											2010-09-25 17:36:58 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  | : live-in? ( vreg bb -- ? ) live-in key? ;
 | 
					
						
							| 
									
										
										
										
											2009-07-22 07:07:28 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-09-25 17:36:58 -04:00
										 |  |  | : live-out? ( vreg bb -- ? ) live-out key? ;
 |