92 lines
3.4 KiB
Scheme
92 lines
3.4 KiB
Scheme
|
;;; unify.ss
|
||
|
;;; Copyright (C) 1996 R. Kent Dybvig
|
||
|
;;; from "The Scheme Programming Language, 2ed" by R. Kent Dybvig
|
||
|
|
||
|
;;; Permission is hereby granted, free of charge, to any person obtaining a
|
||
|
;;; copy of this software and associated documentation files (the "Software"),
|
||
|
;;; to deal in the Software without restriction, including without limitation
|
||
|
;;; the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
||
|
;;; and/or sell copies of the Software, and to permit persons to whom the
|
||
|
;;; Software is furnished to do so, subject to the following conditions:
|
||
|
;;;
|
||
|
;;; The above copyright notice and this permission notice shall be included in
|
||
|
;;; all copies or substantial portions of the Software.
|
||
|
;;;
|
||
|
;;; THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
||
|
;;; IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
||
|
;;; FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
||
|
;;; THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
||
|
;;; LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
||
|
;;; FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
|
||
|
;;; DEALINGS IN THE SOFTWARE.
|
||
|
|
||
|
(define unify #f)
|
||
|
(let ()
|
||
|
;; occurs? returns true if and only if u occurs in v
|
||
|
(define occurs?
|
||
|
(lambda (u v)
|
||
|
(and (pair? v)
|
||
|
(let f ((l (cdr v)))
|
||
|
(and (pair? l)
|
||
|
(or (eq? u (car l))
|
||
|
(occurs? u (car l))
|
||
|
(f (cdr l))))))))
|
||
|
|
||
|
;; sigma returns a new substitution procedure extending s by
|
||
|
;; the substitution of u with v
|
||
|
(define sigma
|
||
|
(lambda (u v s)
|
||
|
(lambda (x)
|
||
|
(let f ((x (s x)))
|
||
|
(if (symbol? x)
|
||
|
(if (eq? x u) v x)
|
||
|
(cons (car x) (map f (cdr x))))))))
|
||
|
|
||
|
;; try-subst tries to substitute u for v but may require a
|
||
|
;; full unification if (s u) is not a variable, and it may
|
||
|
;; fail if it sees that u occurs in v.
|
||
|
(define try-subst
|
||
|
(lambda (u v s ks kf)
|
||
|
(let ((u (s u)))
|
||
|
(if (not (symbol? u))
|
||
|
(uni u v s ks kf)
|
||
|
(let ((v (s v)))
|
||
|
(cond
|
||
|
((eq? u v) (ks s))
|
||
|
((occurs? u v) (kf "cycle"))
|
||
|
(else (ks (sigma u v s)))))))))
|
||
|
|
||
|
;; uni attempts to unify u and v with a continuation-passing
|
||
|
;; style that returns a substitution to the success argument
|
||
|
;; ks or an error message to the failure argument kf. The
|
||
|
;; substitution itself is represented by a procedure from
|
||
|
;; variables to terms.
|
||
|
(define uni
|
||
|
(lambda (u v s ks kf)
|
||
|
(cond
|
||
|
((symbol? u) (try-subst u v s ks kf))
|
||
|
((symbol? v) (try-subst v u s ks kf))
|
||
|
((and (eq? (car u) (car v))
|
||
|
(= (length u) (length v)))
|
||
|
(let f ((u (cdr u)) (v (cdr v)) (s s))
|
||
|
(if (null? u)
|
||
|
(ks s)
|
||
|
(uni (car u)
|
||
|
(car v)
|
||
|
s
|
||
|
(lambda (s) (f (cdr u) (cdr v) s))
|
||
|
kf))))
|
||
|
(else (kf "clash")))))
|
||
|
|
||
|
;; unify shows one possible interface to uni, where the initial
|
||
|
;; substitution is the identity procedure, the initial success
|
||
|
;; continuation returns the unified term, and the initial failure
|
||
|
;; continuation returns the error message.
|
||
|
(set! unify
|
||
|
(lambda (u v)
|
||
|
(uni u
|
||
|
v
|
||
|
(lambda (x) x)
|
||
|
(lambda (s) (s u))
|
||
|
(lambda (msg) msg)))))
|