! Copyright (C) 2004, 2009 Slava Pestov. ! See http://factorcode.org/license.txt for BSD license. USING: fry arrays generic io io.streams.string kernel math namespaces parser sequences strings vectors words quotations effects classes continuations assocs combinators compiler.errors accessors math.order definitions sets hints macros stack-checker.state stack-checker.visitor stack-checker.errors stack-checker.values stack-checker.recursive-state ; IN: stack-checker.backend : push-d ( obj -- ) meta-d push ; : pop-d ( -- obj ) meta-d [ dup 1array #introduce, d-in inc ] [ pop ] if-empty ; : peek-d ( -- obj ) pop-d dup push-d ; : make-values ( n -- values ) [ ] replicate ; : ensure-d ( n -- values ) meta-d 2dup length > [ 2dup [ nip >array ] [ length - make-values ] [ nip delete-all ] 2tri [ length d-in +@ ] [ #introduce, ] [ meta-d push-all ] tri meta-d push-all ] when swap tail* ; : shorten-by ( n seq -- ) [ length swap - ] keep shorten ; inline : consume-d ( n -- seq ) [ ensure-d ] [ meta-d shorten-by ] bi ; : output-d ( values -- ) meta-d push-all ; : produce-d ( n -- values ) make-values dup meta-d push-all ; : push-r ( obj -- ) meta-r push ; : pop-r ( -- obj ) meta-r dup empty? [ too-many-r> ] [ pop ] if ; : consume-r ( n -- seq ) meta-r 2dup length > [ too-many-r> ] when [ swap tail* ] [ shorten-by ] 2bi ; : output-r ( seq -- ) meta-r push-all ; : push-literal ( obj -- ) literals get push ; : pop-literal ( -- rstate obj ) literals get [ pop-d [ 1array #drop, ] [ literal [ recursion>> ] [ value>> ] bi ] bi ] [ pop recursive-state get swap ] if-empty ; : literals-available? ( n -- literals ? ) literals get 2dup length <= [ [ swap tail* ] [ shorten-by ] 2bi t ] [ 2drop f f ] if ; GENERIC: apply-object ( obj -- ) M: wrapper apply-object wrapped>> [ dup word? [ called-dependency depends-on ] [ drop ] if ] [ push-literal ] bi ; M: object apply-object push-literal ; : terminate ( -- ) terminated? on meta-d clone meta-r clone #terminate, ; : check->r ( -- ) meta-r empty? [ too-many->r ] unless ; : infer-quot-here ( quot -- ) meta-r [ V{ } clone \ meta-r set [ apply-object terminated? get not ] all? [ commit-literals check->r ] [ literals get delete-all ] if ] dip \ meta-r set ; : infer-quot ( quot rstate -- ) recursive-state get [ recursive-state set infer-quot-here ] dip recursive-state set ; : time-bomb ( error -- ) '[ _ throw ] infer-quot-here ; : bad-call ( -- ) "call must be given a callable" time-bomb ; : infer-literal-quot ( literal -- ) dup recursive-quotation? [ value>> recursive-quotation-error ] [ dup value>> callable? [ [ value>> ] [ [ recursion>> ] keep add-local-quotation ] bi infer-quot ] [ drop bad-call ] if ] if ; : infer->r ( n -- ) consume-d dup copy-values [ nip output-r ] [ #>r, ] 2bi ; : infer-r> ( n -- ) consume-r dup copy-values [ nip output-d ] [ #r>, ] 2bi ; : consume/produce ( effect quot: ( inputs outputs -- ) -- ) '[ [ in>> length consume-d ] [ out>> length produce-d ] bi @ ] [ terminated?>> [ terminate ] when ] bi ; inline : apply-word/effect ( word effect -- ) swap '[ _ #call, ] consume/produce ; : end-infer ( -- ) meta-d clone #return, ; : required-stack-effect ( word -- effect ) dup stack-effect [ ] [ missing-effect ] ?if ; : infer-word ( word -- ) { { [ dup macro? ] [ do-not-compile ] } { [ dup "no-compile" word-prop ] [ do-not-compile ] } [ dup required-stack-effect apply-word/effect ] } cond ; : with-infer ( quot -- effect visitor ) [ init-inference init-known-values stack-visitor off call end-infer current-effect stack-visitor get ] with-scope ; inline