260 lines
7.2 KiB
Factor
Executable File
260 lines
7.2 KiB
Factor
Executable File
! 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] ;
|