Add ZDD universe-restricted matching operation
author |
Steve Losh <steve@stevelosh.com> |
date |
Tue, 01 Nov 2016 16:38:51 +0000 (2016-11-01) |
parents |
f766019a72af
|
children |
2a71a4230eb3
|
branches/tags |
(none) |
files |
src/zdd.lisp |
Changes
--- a/src/zdd.lisp Tue Nov 01 15:55:14 2016 +0000
+++ b/src/zdd.lisp Tue Nov 01 16:38:51 2016 +0000
@@ -24,6 +24,10 @@
(mapc #'pr (funcall function val))
val)
+(defun line (x)
+ (declare (ignore x))
+ '----------------------------------------)
+
;;;; GraphViz -----------------------------------------------------------------
(setf cl-dot:*dot-path* "/usr/local/bin/dot")
@@ -254,6 +258,10 @@
(leaf nil)))
+(defun zdd-family (&rest sets)
+ (reduce #'zdd-union (mapcar #'zdd-set sets)))
+
+
(defun zdd-keep-supersets-of% (zdd set)
(ematch* (zdd set)
((_ nil) zdd)
@@ -306,8 +314,57 @@
(zdd-keep-avoiders-of% zdd (sort set #'<)))
-(defun zdd-family (&rest sets)
- (reduce #'zdd-union (mapcar #'zdd-set sets)))
+(defun zdd-match% (zdd set lower-bound upper-bound)
+ (recursively ((zdd zdd) (set set))
+ (ematch zdd
+ ;; If Z = ∅, there are no candidates for matching.
+ ((sink nil) (leaf nil))
+
+ ;; If Z = {∅}, the only set ∅ can match is the empty set.
+ ((sink t) (if set
+ (leaf nil)
+ (leaf t)))
+
+ ;; Otherwise Z is a real node.
+ ((node var hi lo)
+ (cond
+ ;; If we're below the lower bound of the universe, just recur down.
+ ((< var lower-bound) (zdd-node var
+ (recur hi set)
+ (recur lo set)))
+
+ ;; If we're above the upper bound of the universe, we're never gonna
+ ;; see anything more we might need to match.
+ ;;
+ ;; If our target set is empty, that's perfect. But if it's NOT empty,
+ ;; we're never gonna satisfy it.
+ ((> var upper-bound) (if set
+ (leaf nil)
+ zdd))
+
+ ;; Otherwise Z's var is within the universe.
+ (t (ematch set
+ ;; If our target is empty, only the lo branch of Z can possibly
+ ;; match.
+ (nil (recur lo set))
+
+ ;; Otherwise we've got a target element. Almost there!
+ ((list* element remaining)
+ (cond
+ ;; If we're below the target element, we recur down the lo
+ ;; branch because the hi branch contains something unwanted.
+ ((< var element) (recur lo set))
+ ;; If we're above the target element, we can never match.
+ ((> var element) (leaf nil))
+ ;; Otherwise, we recur down the hi branch with the rest of our
+ ;; target (the lo branch is always missing this element).
+ ((= var element) (zdd-node var
+ (recur hi remaining)
+ ; jeeeeeeeesus
+ (leaf nil))))))))))))
+
+(defun zdd-match (zdd set lower-bound upper-bound)
+ (zdd-match% zdd (sort set #'<) lower-bound upper-bound))
;;;; Rule Trees ---------------------------------------------------------------
@@ -503,3 +560,22 @@
'(1 5)
'(5)))))
:test #'equal))))
+
+
+(let ((*draw-unique-sinks* nil))
+ (with-zdd
+ (-<> (zdd-family
+ '(1 2 100 200 6000)
+ '(100 200 300)
+ '(99 100 200 300)
+ '(1 9900)
+ '()
+ '(1 2 1001)
+ )
+ (mapprint-through #'enumerate <>)
+ (print-through #'line <>)
+ (zdd-match <> '() 100 999)
+ (mapprint-through #'enumerate <>)
+ (draw <>)
+ (never <>)
+ )))