ikarus/scheme/ikarus.ontology.ss

274 lines
10 KiB
Scheme
Executable File

;;; Ikarus Scheme -- A compiler for R6RS Scheme.
;;; Copyright (C) 2006,2007,2008 Abdulaziz Ghuloum
;;;
;;; This program is free software: you can redistribute it and/or modify
;;; it under the terms of the GNU General Public License version 3 as
;;; published by the Free Software Foundation.
;;;
;;; This program is distributed in the hope that it will be useful, but
;;; WITHOUT ANY WARRANTY; without even the implied warranty of
;;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
;;; General Public License for more details.
;;;
;;; You should have received a copy of the GNU General Public License
;;; along with this program. If not, see <http://www.gnu.org/licenses/>.
(define-syntax define-ontology
(lambda (x)
(define (make-ontology main ls)
(define (set-cons x ls)
(cond
[(memq x ls) ls]
[else (cons x ls)]))
(define (union ls1 ls2)
(cond
[(null? ls1) ls2]
[else (union (cdr ls1) (set-cons (car ls1) ls2))]))
(define (difference ls1 ls2)
(cond
[(null? ls1) '()]
[(memq (car ls1) ls2) (difference (cdr ls1) ls2)]
[else (cons (car ls1) (difference (cdr ls1) ls2))]))
(define (collect-names ls)
(syntax-case ls ()
[() '()]
[((name (of name* ...)) . rest)
(union (cons #'name #'(name* ...)) (collect-names #'rest))]))
(define (expand x all)
(define (lookup x ls)
(cond
[(null? ls) (values 'tag '())]
[else
(let ([a (car ls)])
(cond
[(eq? x (car a))
(values (cadr a) (cdr ls))]
[else
(let-values ([(xp ls) (lookup x (cdr ls))])
(values xp (cons a ls)))]))]))
(let f ([x x] [ls ls])
(let-values ([(xp ls) (lookup x ls)])
(cond
[(pair? xp)
(cons (car xp) (map (lambda (x) (f x ls)) (cdr xp)))]
[(eq? xp 'tag) x]
[else (error 'expand-lookup "invalid" xp)]))))
(define (rename alist x)
(cond
[(symbol? x) (cdr (assq x alist))]
[else (cons (car x) (map (lambda (x) (rename alist x)) (cdr x)))]))
(define (enumerate ls)
(let f ([i 1] [ls ls])
(cond
[(null? ls) '()]
[else (cons i (f (* i 2) (cdr ls)))])))
(define (unique-elements x)
(define (exclude m ls)
(cond
[(null? ls) '()]
[(zero? (bitwise-and m (car ls)))
(cons (car ls) (exclude m (cdr ls)))]
[else (exclude m (cdr ls))]))
(define (exclusive* m* x**)
(cond
[(null? (cdr m*)) (values (car m*) (car x**))]
[else
(let-values ([(m1 x1*) (values (car m*) (car x**))]
[(m2 x2*) (exclusive* (cdr m*) (cdr x**))])
(let ([x1* (exclude m2 x1*)]
[x2* (exclude m1 x2*)])
(values (bitwise-ior m1 m2) (append x1* x2*))))]))
(define (inclusive* m* x**)
(cond
[(null? (cdr m*)) (values (car m*) (car x**))]
[else
(let-values ([(m1 x1*) (values (car m*) (car x**))]
[(m2 x2*) (inclusive* (cdr m*) (cdr x**))])
(values (bitwise-ior m1 m2)
(remp not
(apply append
(map (lambda (x)
(map (lambda (y)
(if (= (bitwise-and m1 m2 x)
(bitwise-and m1 m2 y))
(bitwise-ior x y)
#f))
x2*))
x1*)))))]))
(define (f* ls)
(cond
[(null? ls) (values '() '())]
[else
(let-values ([(m x*) (f (car ls))]
[(m* x**) (f* (cdr ls))])
(values (cons m m*) (cons x* x**)))]))
(define (f x)
(cond
[(integer? x) (values x (list x))]
[else
(let ([tag (car x)] [ls (cdr x)])
(let-values ([(m* x**) (f* ls)])
(case tag
[(exclusive) (exclusive* m* x**)]
[(inclusive) (inclusive* m* x**)]
[else (error 'f "invalid")])))]))
(let-values ([(m ls) (f x)])
ls))
(define (expand-names alist)
(lambda (n)
(let f ([alist alist])
(cond
[(null? alist) '()]
[(zero? (bitwise-and n (cdar alist)))
(f (cdr alist))]
[else
(cons (caar alist) (f (cdr alist)))]))))
(define (extend-alist* ls alist)
(define (extend-alist x alist)
(define (lookup x)
(cond
[(assq x alist) => cdr]
[else (error 'lookup "cannot find" x alist)]))
(let ([name (car x)] [info (cadr x)])
(let ([tag (car info)] [x* (map lookup (cdr info))])
(case tag
[(exclusive)
(cons (cons name (apply bitwise-ior x*)) alist)]
[(inclusive)
(assert (= (apply bitwise-ior x*) (apply bitwise-and x*)))
(cons (cons name (apply bitwise-ior x*)) alist)]
[else (assert #f)]))))
(cond
[(null? ls) alist]
[else
(extend-alist (car ls)
(extend-alist* (cdr ls) alist))]))
(let* ([names (difference (collect-names ls) (map car ls))]
[names-alist (map cons names (enumerate names))])
(let* ([expanded (expand main ls)]
[renamed (rename names-alist expanded)])
(let* ([unique* (list-sort < (unique-elements renamed))]
[canonicals (map (expand-names names-alist) unique*)])
(let* ([canonical-alist (map cons canonicals (enumerate canonicals))]
[seed-alist
(map
(lambda (x)
(let ([ls (filter (lambda (y) (memq x (car y))) canonical-alist)])
(cons x (apply bitwise-ior (map cdr ls)))))
names)])
(extend-alist* ls seed-alist))))))
(define (property-names ls)
(cond
[(null? ls) '()]
[else
(let ([fst (car ls)] [rest (property-names (cdr ls))])
(let ([name (car fst)] [info (cadr fst)])
(case (car info)
[(exclusive) rest]
[(inclusive) (append (cdr info) rest)]
[else (assert #f)])))]))
(define (generate-base-cases T main ls)
(define (value-name x)
(datum->syntax T
(string->symbol
(string-append
(symbol->string (syntax->datum T))
":"
(symbol->string x)))))
(define (predicate-name x)
(datum->syntax T
(string->symbol
(string-append
(symbol->string (syntax->datum T))
":"
(symbol->string x)
"?"))))
(let ([maind (syntax->datum main)] [lsd (syntax->datum ls)])
(let ([alist (make-ontology maind lsd)]
[pnames (property-names lsd)])
(let ([alist (remp (lambda (x) (memq (car x) pnames)) alist)])
(map
(lambda (x) (list (value-name (car x))
(predicate-name (car x))
(cdr x)))
alist)))))
(syntax-case x ()
[(_ T T:description T? T:=? T:and T:or [name cls] [name* cls*] ...)
(with-syntax ([((name* predname* val*) ...)
(generate-base-cases #'T #'name
#'([name cls] [name* cls*] ...))])
#'(begin
(define-record-type (T make-T T?)
(sealed #t)
(fields (immutable n T-n)))
(define (T:and x0 x1)
(make-T (bitwise-and (T-n x0) (T-n x1))))
(define (T:or x0 x1)
(make-T (bitwise-ior (T-n x0) (T-n x1))))
(define (test x v)
(let ([bits (bitwise-and x v)])
(cond
[(= 0 (bitwise-and x v)) 'no]
[(= v (bitwise-ior x v)) 'yes]
[else 'maybe])))
(define name* (make-T val*)) ...
(define (predname* x) (test (T-n x) val*)) ...
(define (T:description x)
(let* ([ls '()]
[ls
(case (predname* x)
[(yes) (cons '(name* yes) ls)]
[else ls])]
...)
ls))
(define (T:=? x y)
(= (T-n x) (T-n y)))
))])))
(define-ontology T T:description T? T=? T:and T:or
[object (inclusive obj-tag obj-immediacy obj-truth)]
[obj-immediacy (exclusive nonimmediate immediate)]
[immediate (exclusive fixnum boolean null char void)]
[obj-truth (exclusive false non-false)]
[obj-tag (exclusive procedure string vector pair null
boolean char number void other-object)]
[boolean (exclusive true false)]
[number (inclusive number-tag number-size number-exactness)]
[number-size (exclusive negative zero positive)]
[number-tag (exclusive fixnum flonum other-number)]
[number-exactness (exclusive exact inexact)]
[exact (exclusive fixnum other-exact)]
[inexact (exclusive flonum other-inexact)]
)
#!eof
(define (do-test expr result expected)
(if (equal? result expected)
(printf "OK: ~s -> ~s\n" expr expected)
(error 'test "failed/got/expected" expr result expected)))
(define-syntax test
(syntax-rules ()
[(_ expr expected) (do-test 'expr expr 'expected)]))
(test (T:object? T:object) yes)
(test (T:object? T:true) yes)
(test (T:true? T:object) maybe)
(test (T:true? T:true) yes)
(test (T:true? T:false) no)
(test (T:true? T:null) no)
(test (T:non-false? T:true) yes)
(test (T:non-false? T:null) yes)
(test (T:non-false? T:false) no)
(test (T:non-false? T:boolean) maybe)
(test (T:non-false? T:object) maybe)
(test (T:boolean? T:true) yes)
(test (T:boolean? T:false) yes)
(test (T:boolean? (T:or T:true T:false)) yes)
(test (T:boolean? (T:and T:true T:false)) no)
(test (T:object? (T:and T:true T:false)) no)