factor/extra/inverse/inverse.factor

232 lines
6.4 KiB
Factor
Raw Normal View History

2007-09-20 18:09:08 -04:00
USING: kernel words inspector slots quotations sequences assocs
math arrays inference effects shuffle continuations debugger
tuples namespaces vectors bit-arrays byte-arrays strings sbufs
math.functions macros ;
IN: inverse
TUPLE: fail ;
: fail ( -- * ) \ fail construct-empty 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 ;
2007-11-21 17:56:28 -05:00
: define-math-inverse ( word quot1 quot2 -- )
2array "math-inverse" set-word-prop ;
2007-11-22 00:43:30 -05:00
: define-pop-inverse ( word n quot -- )
>r dupd "pop-length" set-word-prop r>
"pop-inverse" set-word-prop ;
2007-09-20 18:09:08 -04:00
TUPLE: no-inverse word ;
: no-inverse ( word -- * ) \ no-inverse construct-empty throw ;
M: no-inverse summary
drop "The word cannot be used in pattern matching" ;
2007-11-21 17:56:28 -05:00
: next ( revquot -- revquot* first )
dup empty?
[ "Badly formed math inverse" throw ]
[ unclip-slice ] if ;
: constant-word? ( word -- ? )
stack-effect
[ effect-out length 1 = ] keep
effect-in length 0 = and ;
2007-09-20 18:09:08 -04:00
2007-11-21 17:56:28 -05:00
: assure-constant ( constant -- quot )
2007-11-22 00:43:30 -05:00
dup word? [
dup constant-word?
[ "Badly formed math inverse" throw ] unless
] when 1quotation ;
2007-09-20 18:09:08 -04:00
2007-11-21 17:56:28 -05:00
: swap-inverse ( math-inverse revquot -- revquot* quot )
2007-11-22 00:43:30 -05:00
next assure-constant rot second [ swap ] swap 3compose ;
2007-09-20 18:09:08 -04:00
2007-11-21 17:56:28 -05:00
: 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 ;
2007-09-20 18:09:08 -04:00
2007-11-22 00:43:30 -05:00
GENERIC: inverse ( revquot word -- revquot* quot )
2007-11-28 15:33:58 -05:00
DEFER: [undo]
2007-11-22 00:43:30 -05:00
M: word inverse
dup "inverse" word-prop [ ]
2007-11-28 15:33:58 -05:00
[ dup primitive? [ no-inverse ] [ word-def [undo] ] if ] ?if ;
2007-11-22 00:43:30 -05:00
: undo-literal ( object -- quot )
[ =/fail ] curry ;
M: object inverse undo-literal ;
M: symbol inverse undo-literal ;
PREDICATE: word math-inverse "math-inverse" word-prop ;
M: math-inverse inverse
"math-inverse" word-prop
swap next dup \ swap =
[ drop swap-inverse ] [ pull-inverse ] if ;
PREDICATE: word pop-inverse "pop-length" word-prop ;
M: pop-inverse inverse
[ "pop-length" word-prop cut-slice swap ] keep
"pop-inverse" word-prop compose call ;
2007-11-21 17:56:28 -05:00
: (undo) ( revquot -- )
2007-11-22 00:43:30 -05:00
dup empty? [ drop ]
[ unclip-slice inverse % (undo) ] if ;
2007-09-20 18:09:08 -04:00
: [undo] ( quot -- undo )
2007-11-21 17:56:28 -05:00
reverse [ (undo) ] [ ] make ;
2007-09-20 18:09:08 -04:00
MACRO: undo ( quot -- ) [undo] ;
2007-11-28 15:33:58 -05:00
! Inverse of selected words
2007-09-20 18:09:08 -04:00
\ 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 ;
2007-11-21 17:56:28 -05:00
\ + [ - ] [ - ] define-math-inverse
\ - [ + ] [ - ] define-math-inverse
\ * [ / ] [ / ] define-math-inverse
\ / [ * ] [ / ] define-math-inverse
\ ^ [ recip ^ ] [ [ log ] 2apply / ] define-math-inverse
2007-09-20 18:09:08 -04:00
\ ? 2 [
[ assert-literal ] 2apply
[ swap >r over = r> swap [ 2drop f ] [ = [ t ] [ fail ] if ] if ]
2curry
] define-pop-inverse
: _ f ;
\ _ [ 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
! Constructor inverse
: deconstruct-pred ( class -- quot )
"predicate" word-prop [ dupd call assure ] curry ;
: slot-readers ( class -- quot )
"slots" word-prop 1 tail ! tail gets rid of delegate
[ slot-spec-reader 1quotation [ keep ] curry ] map concat
2007-11-22 00:43:30 -05:00
[ ] like [ drop ] compose ;
2007-09-20 18:09:08 -04:00
: ?wrapped ( object -- wrapped )
dup wrapper? [ wrapped ] when ;
: boa-inverse ( class -- quot )
2007-11-22 00:43:30 -05:00
[ deconstruct-pred ] keep slot-readers compose ;
2007-09-20 18:09:08 -04:00
\ construct-boa 1 [ ?wrapped boa-inverse ] define-pop-inverse
: empty-inverse ( class -- quot )
deconstruct-pred
[ tuple>array 1 tail [ ] contains? [ fail ] when ]
compose ;
\ construct-empty 1 [ ?wrapped empty-inverse ] define-pop-inverse
: writer>reader ( word -- word' )
[ "writing" word-prop "slots" word-prop ] keep
[ swap slot-spec-writer = ] curry find nip slot-spec-reader ;
: construct-inverse ( class setters -- quot )
>r deconstruct-pred r>
[ writer>reader ] map [ get-slots ] curry
compose ;
2007-11-22 00:43:30 -05:00
\ construct 2 [ >r ?wrapped r> construct-inverse ] define-pop-inverse
2007-09-20 18:09:08 -04:00
! More useful inverse-based combinators
: recover-fail ( try fail -- )
[ drop call ] [
>r nip r> dup fail?
[ drop call ] [ nip throw ] if
] recover ; inline
2007-11-22 00:43:30 -05:00
: true-out ( quot effect -- quot' )
effect-out [ ndrop ] curry
[ t ] 3compose ;
2007-09-20 18:09:08 -04:00
2007-11-22 00:43:30 -05:00
: false-recover ( effect -- quot )
effect-in [ ndrop f ] curry [ recover-fail ] curry ;
: [matches?] ( quot -- undoes?-quot )
[undo] dup infer [ true-out ] keep false-recover curry ;
MACRO: matches? ( quot -- ? ) [matches?] ;
2007-09-20 18:09:08 -04:00
TUPLE: no-match ;
: no-match ( -- * ) \ no-match construct-empty throw ;
2007-11-22 00:43:30 -05:00
M: no-match summary drop "Fall through in switch" ;
2007-09-20 18:09:08 -04:00
: recover-chain ( seq -- quot )
[ no-match ] [ swap \ recover-fail 3array >quotation ] reduce ;
2007-11-22 00:43:30 -05:00
: [switch] ( quot-alist -- quot )
reverse [ >r [undo] r> compose ] { } assoc>map
2007-09-20 18:09:08 -04:00
recover-chain ;
2007-11-22 00:43:30 -05:00
MACRO: switch ( quot-alist -- ) [switch] ;