! Copyright (C) 2007, 2008 Daniel Ehrenberg. ! See http://factorcode.org/license.txt for BSD license. USING: accessors kernel words summary slots quotations sequences assocs math arrays stack-checker effects generalizations continuations debugger classes.tuple namespaces make vectors bit-arrays byte-arrays strings sbufs math.functions macros sequences.private combinators mirrors combinators.short-circuit ; IN: inverse TUPLE: fail ; : fail ( -- * ) \ fail new throw ; M: fail summary drop "Unification failed" ; : assure ( ? -- ) [ fail ] unless ; : =/fail ( obj1 obj2 -- ) = assure ; ! Inverse of a quotation : define-inverse ( word quot -- ) "inverse" set-word-prop ; : define-math-inverse ( word quot1 quot2 -- ) pick 1quotation 3array "math-inverse" set-word-prop ; : define-pop-inverse ( word n quot -- ) >r dupd "pop-length" set-word-prop r> "pop-inverse" set-word-prop ; TUPLE: no-inverse word ; : no-inverse ( word -- * ) \ no-inverse new throw ; M: no-inverse summary drop "The word cannot be used in pattern matching" ; : next ( revquot -- revquot* first ) [ "Badly formed math inverse" throw ] [ unclip-slice ] if-empty ; : constant-word? ( word -- ? ) stack-effect [ out>> length 1 = ] keep in>> length 0 = and ; : assure-constant ( constant -- quot ) dup word? [ "Badly formed math inverse" throw ] when 1quotation ; : swap-inverse ( math-inverse revquot -- revquot* quot ) next assure-constant rot second [ swap ] swap 3compose ; : pull-inverse ( math-inverse revquot const -- revquot* quot ) assure-constant rot first compose ; : ?word-prop ( word/object name -- value/f ) over word? [ word-prop ] [ 2drop f ] if ; : undo-literal ( object -- quot ) [ =/fail ] curry ; PREDICATE: normal-inverse < word "inverse" word-prop ; PREDICATE: math-inverse < word "math-inverse" word-prop ; PREDICATE: pop-inverse < word "pop-length" word-prop ; UNION: explicit-inverse normal-inverse math-inverse pop-inverse ; : enough? ( stack word -- ? ) dup deferred? [ 2drop f ] [ [ >r length r> 1quotation infer in>> >= ] [ 3drop f ] recover ] if ; : fold-word ( stack word -- stack ) 2dup enough? [ 1quotation with-datastack ] [ >r % r> , { } ] if ; : fold ( quot -- folded-quot ) [ { } swap [ fold-word ] each % ] [ ] make ; : flattenable? ( object -- ? ) { [ word? ] [ primitive? not ] [ { "inverse" "math-inverse" "pop-inverse" } [ word-prop ] with contains? not ] } 1&& ; : (flatten) ( quot -- ) [ dup flattenable? [ def>> (flatten) ] [ , ] if ] each ; : retain-stack-overflow? ( error -- ? ) { "kernel-error" 14 f f } = ; : flatten ( quot -- expanded ) [ [ (flatten) ] [ ] make ] [ dup retain-stack-overflow? [ drop "No inverse defined on recursive word" ] when throw ] recover ; GENERIC: inverse ( revquot word -- revquot* quot ) M: object inverse undo-literal ; M: symbol inverse undo-literal ; M: word inverse drop "Inverse is undefined" throw ; M: normal-inverse inverse "inverse" word-prop ; M: math-inverse inverse "math-inverse" word-prop swap next dup \ swap = [ drop swap-inverse ] [ pull-inverse ] if ; M: pop-inverse inverse [ "pop-length" word-prop cut-slice swap >quotation ] keep "pop-inverse" word-prop compose call ; : (undo) ( revquot -- ) [ unclip-slice inverse % (undo) ] unless-empty ; : [undo] ( quot -- undo ) flatten fold reverse [ (undo) ] [ ] make ; MACRO: undo ( quot -- ) [undo] ; ! Inverse of selected words \ swap [ swap ] define-inverse \ dup [ [ =/fail ] keep ] define-inverse \ 2dup [ over =/fail over =/fail ] define-inverse \ 3dup [ pick =/fail pick =/fail pick =/fail ] define-inverse \ pick [ >r pick r> =/fail ] define-inverse \ tuck [ swapd [ =/fail ] keep ] define-inverse \ >r [ r> ] define-inverse \ r> [ >r ] define-inverse \ tuple>array [ >tuple ] define-inverse \ >tuple [ tuple>array ] define-inverse \ reverse [ reverse ] define-inverse \ undo 1 [ [ call ] curry ] define-pop-inverse \ map 1 [ [undo] [ over sequence? assure map ] curry ] define-pop-inverse \ exp [ log ] define-inverse \ log [ exp ] define-inverse \ not [ not ] define-inverse \ sq [ sqrt ] define-inverse \ sqrt [ sq ] define-inverse : assert-literal ( n -- n ) dup [ word? ] keep symbol? not and [ "Literal missing in pattern matching" throw ] when ; \ + [ - ] [ - ] define-math-inverse \ - [ + ] [ - ] define-math-inverse \ * [ / ] [ / ] define-math-inverse \ / [ * ] [ / ] define-math-inverse \ ^ [ recip ^ ] [ [ log ] bi@ / ] define-math-inverse \ ? 2 [ [ assert-literal ] bi@ [ swap >r over = r> swap [ 2drop f ] [ = [ t ] [ fail ] if ] if ] 2curry ] define-pop-inverse DEFER: _ \ _ [ drop ] define-inverse : both ( object object -- object ) dupd assert= ; \ both [ dup ] define-inverse : assure-length ( seq length -- seq ) over length =/fail ; { { >array array? } { >vector vector? } { >fixnum fixnum? } { >bignum bignum? } { >bit-array bit-array? } { >float float? } { >byte-array byte-array? } { >string string? } { >sbuf sbuf? } { >quotation quotation? } } [ \ dup swap \ assure 3array >quotation define-inverse ] assoc-each ! These actually work on all seqs--should they? \ 1array [ 1 assure-length first ] define-inverse \ 2array [ 2 assure-length first2 ] define-inverse \ 3array [ 3 assure-length first3 ] define-inverse \ 4array [ 4 assure-length first4 ] define-inverse \ first [ 1array ] define-inverse \ first2 [ 2array ] define-inverse \ first3 [ 3array ] define-inverse \ first4 [ 4array ] define-inverse \ prefix [ unclip ] define-inverse \ unclip [ prefix ] define-inverse \ suffix [ dup but-last swap peek ] define-inverse ! Constructor inverse : deconstruct-pred ( class -- quot ) "predicate" word-prop [ dupd call assure ] curry ; : slot-readers ( class -- quot ) all-slots [ name>> reader-word 1quotation [ keep ] curry ] map concat [ ] like [ drop ] compose ; : ?wrapped ( object -- wrapped ) dup wrapper? [ wrapped>> ] when ; : boa-inverse ( class -- quot ) [ deconstruct-pred ] keep slot-readers compose ; \ boa 1 [ ?wrapped boa-inverse ] define-pop-inverse : empty-inverse ( class -- quot ) deconstruct-pred [ tuple>array rest [ ] contains? [ fail ] when ] compose ; \ new 1 [ ?wrapped empty-inverse ] define-pop-inverse ! More useful inverse-based combinators : recover-fail ( try fail -- ) [ drop call ] [ >r nip r> dup fail? [ drop call ] [ nip throw ] if ] recover ; inline : true-out ( quot effect -- quot' ) out>> [ ndrop ] curry [ t ] 3compose ; : false-recover ( effect -- quot ) in>> [ ndrop f ] curry [ recover-fail ] curry ; : [matches?] ( quot -- undoes?-quot ) [undo] dup infer [ true-out ] keep false-recover curry ; MACRO: matches? ( quot -- ? ) [matches?] ; TUPLE: no-match ; : no-match ( -- * ) \ no-match new throw ; M: no-match summary drop "Fall through in switch" ; : recover-chain ( seq -- quot ) [ no-match ] [ swap \ recover-fail 3array >quotation ] reduce ; : [switch] ( quot-alist -- quot ) [ dup quotation? [ [ ] swap 2array ] when ] map reverse [ >r [undo] r> compose ] { } assoc>map recover-chain ; MACRO: switch ( quot-alist -- ) [switch] ;