| 
									
										
										
										
											2009-02-11 14:56:09 -05:00
										 |  |  | ! Copyright (C) 2009 Philipp Brüschweiler | 
					
						
							|  |  |  | ! See http://factorcode.org/license.txt for BSD license. | 
					
						
							| 
									
										
										
										
											2009-02-07 19:03:35 -05:00
										 |  |  | USING: accessors assocs combinators combinators.short-circuit | 
					
						
							|  |  |  | effects fry infix.parser infix.ast kernel locals.parser | 
					
						
							|  |  |  | locals.types math multiline namespaces parser quotations | 
					
						
							| 
									
										
										
										
											2009-05-15 00:23:06 -04:00
										 |  |  | sequences summary words vocabs.parser ;
 | 
					
						
							| 
									
										
										
										
											2009-02-07 19:03:35 -05:00
										 |  |  | IN: infix | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | <PRIVATE
 | 
					
						
							|  |  |  | : prepare-operand ( term -- quot )
 | 
					
						
							|  |  |  |     dup callable? [ 1quotation ] unless ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ERROR: local-not-defined name ;
 | 
					
						
							|  |  |  | M: local-not-defined summary | 
					
						
							|  |  |  |     drop "local is not defined" ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : >local-word ( string -- word )
 | 
					
						
							| 
									
										
										
										
											2009-02-22 18:52:04 -05:00
										 |  |  |     locals get ?at [ local-not-defined ] unless ;
 | 
					
						
							| 
									
										
										
										
											2009-02-07 19:03:35 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | : select-op ( string -- word )
 | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         { "+" [ [ + ] ] } | 
					
						
							|  |  |  |         { "-" [ [ - ] ] } | 
					
						
							|  |  |  |         { "*" [ [ * ] ] } | 
					
						
							|  |  |  |         { "/" [ [ / ] ] } | 
					
						
							|  |  |  |         [ drop [ mod ] ] | 
					
						
							|  |  |  |     } case ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | GENERIC: infix-codegen ( ast -- quot/number )
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: ast-number infix-codegen value>> ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: ast-local infix-codegen | 
					
						
							|  |  |  |     name>> >local-word ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: ast-array infix-codegen | 
					
						
							|  |  |  |     [ index>> infix-codegen prepare-operand ] | 
					
						
							|  |  |  |     [ name>> >local-word ] bi '[ @ _ nth ] ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: ast-op infix-codegen | 
					
						
							|  |  |  |     [ left>> infix-codegen ] [ right>> infix-codegen ] | 
					
						
							|  |  |  |     [ op>> select-op ] tri
 | 
					
						
							| 
									
										
										
										
											2009-04-15 20:03:44 -04:00
										 |  |  |     2over [ number? ] both? [ call( a b -- c ) ] [ | 
					
						
							| 
									
										
										
										
											2009-02-07 19:03:35 -05:00
										 |  |  |         [ [ prepare-operand ] bi@ ] dip '[ @ @ @ ] | 
					
						
							|  |  |  |     ] if ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: ast-negation infix-codegen | 
					
						
							|  |  |  |     term>> infix-codegen | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         { [ dup number? ] [ neg ] } | 
					
						
							|  |  |  |         { [ dup callable? ] [ '[ @ neg ] ] } | 
					
						
							|  |  |  |         [ '[ _ neg ] ] ! local word | 
					
						
							|  |  |  |     } cond ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ERROR: bad-stack-effect word ;
 | 
					
						
							|  |  |  | M: bad-stack-effect summary | 
					
						
							|  |  |  |     drop "Words used in infix must declare a stack effect and return exactly one value" ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : check-word ( argcount word -- ? )
 | 
					
						
							|  |  |  |     dup stack-effect [ ] [ bad-stack-effect ] ?if
 | 
					
						
							|  |  |  |     [ in>> length ] [ out>> length ] bi
 | 
					
						
							|  |  |  |     [ = ] dip 1 = and ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : find-and-check ( args argcount string -- quot )
 | 
					
						
							|  |  |  |     dup search [ ] [ no-word ] ?if
 | 
					
						
							|  |  |  |     [ nip ] [ check-word ] 2bi
 | 
					
						
							|  |  |  |     [ 1quotation compose ] [ bad-stack-effect ] if ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : arguments-codegen ( seq -- quot )
 | 
					
						
							|  |  |  |     dup empty? [ drop [ ] ] [ | 
					
						
							|  |  |  |         [ infix-codegen prepare-operand ] | 
					
						
							|  |  |  |         [ compose ] map-reduce
 | 
					
						
							|  |  |  |     ] if ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | M: ast-function infix-codegen | 
					
						
							|  |  |  |     [ arguments>> [ arguments-codegen ] [ length ] bi ] | 
					
						
							|  |  |  |     [ name>> ] bi find-and-check ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | : [infix-parse ( end -- result/quot )
 | 
					
						
							|  |  |  |     parse-multiline-string build-infix-ast | 
					
						
							|  |  |  |     infix-codegen prepare-operand ;
 | 
					
						
							|  |  |  | PRIVATE>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-21 02:27:50 -04:00
										 |  |  | SYNTAX: [infix | 
					
						
							|  |  |  |     "infix]" [infix-parse parsed \ call parsed ;
 | 
					
						
							| 
									
										
										
										
											2009-02-07 19:03:35 -05:00
										 |  |  | 
 | 
					
						
							|  |  |  | <PRIVATE
 | 
					
						
							| 
									
										
										
										
											2009-05-14 23:31:29 -04:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 19:03:35 -05:00
										 |  |  | : parse-infix-locals ( assoc end -- quot )
 | 
					
						
							| 
									
										
										
										
											2009-05-14 23:31:29 -04:00
										 |  |  |     '[ _ [infix-parse prepare-operand ] ((parse-lambda)) ;
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-07 19:03:35 -05:00
										 |  |  | PRIVATE>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-21 02:27:50 -04:00
										 |  |  | SYNTAX: [infix| | 
					
						
							| 
									
										
										
										
											2009-02-07 19:03:35 -05:00
										 |  |  |     "|" parse-bindings "infix]" parse-infix-locals <let> | 
					
						
							| 
									
										
										
										
											2009-03-21 02:27:50 -04:00
										 |  |  |     ?rewrite-closures over push-all ;
 |