/usr/share/acl2-7.2dfsg/books/tools/defined-const.lisp is in acl2-books-source 7.2dfsg-3.
This file is owned by root:root, with mode 0o644.
The actual contents of the file can be viewed below.
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 | (in-package "ACL2")
;; This book defines an event-introducing macro, DEFINED-CONST. This only
;; really works with the HONS system (in particular, memoize.)
;; Usage:
;; (defined-const *my-result* (my-expensive (ground-term))
;; This produces the following events:
;; (defconst *my-result* <result>)
;; (defthm *my-result*-def
;; (equal (my-expensive (ground-term))
;; *my-result*))
;; where <result> is the value of (my-expensive (ground-term)).
;; The tricky thing that DEFINED-CONST automates is doing this while only
;; running (my-expensive (ground-term)) once.
;; Optional keyword arguments are :thmname (chooses the name of the defthm)
;; and :rule-classes (chooses the rule-classes of the defthm; default is
;; :rewrite.)
;; Note that if you use :rule-classes :rewrite and want the theorem to fire,
;; you'll need to disable the appropriate executable counterparts of functions
;; that appear in the term. Also, ACL2 may refuse to create rewrite rules
;; targetting certaing ground expressions; Matt Kaufmann provided this example
;; which fails:
;; (defined-const *foo* (+ 2 3))
;; In these cases you can supply :rule-classes nil to allow this to work.
(defun defined-const-fn (constname
term
thmname
rule-classes
evisc hons)
(let ((thmname (or thmname (intern-in-package-of-symbol
(concatenate 'string
(symbol-name constname) "-DEF")
constname))))
`(with-output
:off :all :on (error)
(encapsulate nil
(local
(progn
(in-theory '((:definition hons-copy)))
(defun defined-const-memoize-fn1 ()
(declare (xargs :verify-guards nil))
,term)
(in-theory (disable (defined-const-memoize-fn1)))
(defun defined-const-memoize-fn ()
(declare (xargs :guard t))
,(if hons
'(hons-copy (ec-call (defined-const-memoize-fn1)))
'(ec-call (defined-const-memoize-fn1))))
(defthm defined-const-memoize-fn-is-term
(equal ,term
(defined-const-memoize-fn))
:hints (("goal" :in-theory (disable (defined-const-memoize-fn))))
:rule-classes nil)
#+hons
(memoize 'defined-const-memoize-fn)))
(make-event
`(make-event
(let ((val ,(if (eq (getprop 'defined-const-memoize-fn 'formals 'none 'current-acl2-world
(w state))
'none)
;; This checks to see whether the local events above
;; were run. If so, we run defined-const-memoize-fn;
;; if not, it probably means we're in an include-book
;; of an uncertified book, in which case we just run
;; the original term.
',term
'(defined-const-memoize-fn))))
`(progn (with-output
:stack :pop
(defconst ,',',constname
,,,(if hons
```(hons-copy ',val)
```',val)))
(with-output
:stack :pop
(defthm ,',',thmname
(equal ,',',term ,',',constname)
:hints (("goal" :use defined-const-memoize-fn-is-term))
:rule-classes ,',',rule-classes))
(with-output
:stack :pop
(table defined-const-table
',',',constname ',',',thmname))
,@(and ,,evisc
`((with-output
:stack :pop
(table evisc-table ,',',constname
,(let ((name (symbol-name ',',constname)))
(if (may-need-slashes name)
(concatenate 'string "#.|" name "|")
(concatenate 'string "#."
name)))))))
,@(and ,,hons
`((with-output
:stack :pop
(table persistent-hons-table
,',',constname t))))))))))))
(defmacro defined-const (constname
term
&key
thmname
rule-classes
evisc hons)
(defined-const-fn constname term thmname rule-classes evisc hons))
;; Simple tests to make sure it works.
(local (defined-const *foo* 3))
(local (defined-const *bar* 3 :hons t))
(local (defined-const *baz* 3 :evisc t :hons t))
|