From 3e11a7f2040d89271113fcc5ffe2cbebd43afe52 Mon Sep 17 00:00:00 2001 From: Slava Pestov Date: Wed, 18 Jun 2008 19:46:37 -0500 Subject: [PATCH] Debugging persistent vectors --- .../persistent-vectors-tests.factor | 4 +++ .../persistent-vectors.factor | 33 ++++++++++++------- 2 files changed, 25 insertions(+), 12 deletions(-) diff --git a/extra/persistent-vectors/persistent-vectors-tests.factor b/extra/persistent-vectors/persistent-vectors-tests.factor index a4e4ad33fe..45eb894e67 100644 --- a/extra/persistent-vectors/persistent-vectors-tests.factor +++ b/extra/persistent-vectors/persistent-vectors-tests.factor @@ -48,6 +48,10 @@ random namespaces vectors math math.order ; [ ] [ PV{ } "1" set ] unit-test [ ] [ V{ } clone "2" set ] unit-test +: push/pop-test ( vec -- vec' ) 3 swap ppush 3 swap ppush ppop ; + +[ ] [ PV{ } 10000 [ push/pop-test ] times drop ] unit-test + [ t ] [ 100 [ drop diff --git a/extra/persistent-vectors/persistent-vectors.factor b/extra/persistent-vectors/persistent-vectors.factor index f9f4b68933..c80de3b0cd 100644 --- a/extra/persistent-vectors/persistent-vectors.factor +++ b/extra/persistent-vectors/persistent-vectors.factor @@ -1,7 +1,7 @@ ! Based on Clojure's PersistentVector by Rich Hickey. USING: math accessors kernel sequences.private sequences arrays -combinators parser prettyprint.backend ; +combinators combinators.lib parser prettyprint.backend ; IN: persistent-vectors ERROR: empty-error pvec ; @@ -123,30 +123,39 @@ M: persistent-vector new-nth ( obj i pvec -- pvec' ) ] if ] if ; +: ppop-tail ( pvec -- pvec' ) + [ clone [ ppop ] change-children ] change-tail ; + : (ppop-contraction) ( node -- node' tail' ) clone [ unclip-last swap ] change-children swap ; : ppop-contraction ( node -- node' tail' ) - [ (ppop-contraction) ] [ level>> 1 = ] bi swap and ; + dup children>> length 1 = + [ children>> peek f swap ] + [ (ppop-contraction) ] + if ; : (ppop-new-tail) ( root -- root' tail' ) dup level>> 1 > [ - dup children>> peek (ppop-new-tail) over children>> empty? - [ 2drop ppop-contraction ] [ [ swap node-set-last ] dip ] if + dup children>> peek (ppop-new-tail) over + [ [ swap node-set-last ] dip ] + [ 2drop ppop-contraction ] + if ] [ ppop-contraction ] if ; -: ppop-tail ( pvec -- pvec' ) - [ clone [ ppop ] change-children ] change-tail ; +: trivial? ( node -- ? ) + { [ level>> 1 > ] [ children>> length 1 = ] } 1&& ; : ppop-new-tail ( pvec -- pvec' ) - dup root>> (ppop-new-tail) - [ - dup [ level>> 1 > ] [ children>> length 1 = ] bi and - [ children>> first ] when - ] dip - [ >>root ] [ >>tail ] bi* ; + dup root>> (ppop-new-tail) [ + { + { [ dup not ] [ drop T{ node f { } 1 } ] } + { [ dup trivial? ] [ children>> first ] } + [ ] + } cond + ] dip [ >>root ] [ >>tail ] bi* ; PRIVATE>