diff --git a/extra/lisp/lisp.factor b/extra/lisp/lisp.factor index d2de6fe278..be43d50a18 100644 --- a/extra/lisp/lisp.factor +++ b/extra/lisp/lisp.factor @@ -1,29 +1,38 @@ ! Copyright (C) 2008 James Cash ! See http://factorcode.org/license.txt for BSD license. USING: kernel peg.ebnf peg.expr math.parser sequences arrays strings combinators.lib -namespaces combinators math bake ; +namespaces combinators math bake locals.private accessors vectors syntax ; IN: lisp TUPLE: lisp-symbol name ; -C: lisp-symbol +C: lisp-symbol EBNF: lisp-expr -_ = (" " | "\t" | "\n")* -LPAREN = "(" -RPAREN = ")" -digit = [0-9] -integer = (digit)+ => [[ string>number ]] -float = (digit)+ "." (digit)* => [[ first3 >string [ >string ] 2 ndip 3append string>number ]] -number = float - | integer -identifier = [a-zA-Z] ([^(){} ])* => [[ [ 1 head ] [ second ] bi append >string ]] -atom = number - | identifier -list-item = _ (atom|list) _ => [[ second ]] -list = LPAREN (list-item)* RPAREN => [[ second ]] +_ = (" " | "\t" | "\n")* +LPAREN = "(" +RPAREN = ")" +dquote = '"' +digit = [0-9] +integer = (digit)+ => [[ string>number ]] +float = (digit)+ "." (digit)* => [[ first3 >string [ >string ] dipd 3append string>number ]] +number = float + | integer +id-specials = "!" | "$" | "%" | "&" | "*" | "/" | ":" | "<" + | " =" | ">" | "?" | "^" | "_" | "~" | "+" | "-" | "." | "@" +letters = [a-zA-Z] => [[ 1array >string ]] +initials = letters | id-specials +numbers = [0-9] => [[ 1array >string ]] +subsequents = initials | numbers +identifier = initials (subsequents)* => [[ first2 concat append ]] +string = dquote ("\" . | !(dquote) . )* dquote => [[ second >string ]] +atom = number + | identifier + | string +list-item = _ (atom|list) _ => [[ second ]] +list = LPAREN (list-item)* RPAREN => [[ second ]] ;EBNF - + DEFER: convert-form : convert-body ( lisp-form -- quot ) @@ -32,12 +41,31 @@ DEFER: convert-form : convert-if ( lisp-form -- quot ) 1 tail [ convert-form ] map reverse first3 [ % , , if ] bake ; +: convert-begin ( lisp-form -- quot ) + 1 tail convert-body ; + +: convert-cond ( lisp-form -- quot ) + 1 tail [ [ convert-body map ] map ] [ % cond ] bake ; + : convert-general-form ( lisp-form -- quot ) unclip swap convert-body [ % , ] bake ; - + +> swap member? [ name>> make-local ] [ ] if ] + [ dup vector? [ localize-body ] [ nip ] if ] if ] with map ; inline +PRIVATE> + +: convert-lambda ( lisp-form -- quot ) + 1 tail unclip reverse [ name>> ] map dup make-locals dup push-locals + [ swap localize-body convert-body ] dipd pop-locals swap ; + : convert-list-form ( lisp-form -- quot ) dup first - { { [ dup "if" equal? ] [ drop convert-if ] } + { { [ dup "if" equal? ] [ drop convert-if ] } + { [ dup "begin" equal? ] [ drop convert-begin ] } + { [ dup "cond" equal? ] [ drop convert-cond ] } + { [ dup "lambda" equal? ] [ drop convert-lambda ] } [ drop convert-general-form ] } cond ;