This file is indexed.

/usr/share/acl2-7.2dfsg/books/make-event/defrule.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
; Copyright (C) 2013, Regents of the University of Texas
; Written by Matt Kaufmann, October, 2010
; License: A 3-clause BSD license.  See the LICENSE file distributed with ACL2.

; This book introduces a macro, DEFRULE.  DEFRULE behaves just like defthm,
; except that its intended use is to generate rewrite rules from formulas that
; may involve calls of IF, or of macros like COND, CASE, AND, and OR whose
; expansions introduce IF expressions.

; Note to those who are learning to use MAKE-EVENT to create events by
; expanding macro calls (really, an apology for using a system-level function,
; and an explanation of how one can find such functions):

; Notice the call below of TRANSLATE.  While TRANSLATE is a "system-level"
; function (used in the ACL2 implementation) which therefore doesn't have a
; documentation topic, one can find it in the ACL2 source code with documention
; in the form of comments.  We figured out how to call TRANSLATE below by
; looking up the definition of DEFTHM in the ACL2 sources, which led to
; DEFTHM-FN and then DEFTHM-FN1, which contained a call of TRANSLATE that was
; easily adapted for the present purpose.

(in-package "ACL2")

(local ; just a lemma needed for guard verification below
 (defthm true-listp-revappend
   (equal (true-listp (revappend x y))
          (true-listp y))))

(verify-termination dumb-negate-lit) ; and guards

(defun parse-if-as-implies1 (x hyps)
  (declare (xargs :guard (and (pseudo-termp x)
                              (true-listp hyps))))
  (case-match x
    (('if test tbr ''t) ; Note that 't is the translated form of t.
     (parse-if-as-implies1 tbr (cons test hyps)))
    (('if test test fbr) ; (or test test, Boolean equivalent to (if test t fbr)
     (parse-if-as-implies1 fbr
                           (cons (dumb-negate-lit test)
                                 hyps)))
    (&
     (cond ((cdr hyps) `(implies (and ,@(reverse hyps)) ,x))
           (hyps `(implies ,(car hyps) ,x))
           (t x)))))

(defun parse-if-as-implies (x)
  (declare (xargs :guard (pseudo-termp x)))
  (parse-if-as-implies1 x nil))

(defmacro defrule (name form &rest rest)

; Use this exactly like defthm, but the top-level translated IF structure is
; replaced by IMPLIES calls in order to generate a suitable form for a rewrite
; rule.

  `(make-event
    (er-let*
        ((tform ; mimic the call in source function defthm-fn1
          (translate ',form t t t '(defrule . ,name) (w state) state)))
      (value (list* 'defthm ',name
                    (untranslate (parse-if-as-implies tform) t
                                 (w state))
                    ',rest)))))

; Example:
(local
 (defrule my-test
   (or (acl2-numberp x)
       (acl2-numberp y)
       (equal (+ x y) 0))))

; Check (apologies for use of low-level system functions):
(assert-event
 (equal (getprop 'my-test 'untranslated-theorem nil 'current-acl2-world
                 (w state))
        '(IMPLIES (AND (NOT (ACL2-NUMBERP X))
                       (NOT (ACL2-NUMBERP Y)))
                  (EQUAL (+ X Y) 0))))