Filtering...

universal-equiv

books/centaur/misc/universal-equiv

Included Books

other
(in-package "ACL2")
include-book
(include-book "clause-processors/equality" :dir :system)
include-book
(include-book "xdoc/top" :dir :system)
other
(set-state-ok t)
other
(defsection def-universal-equiv
  :parents (macro-libraries)
  :short "A macro for defining universally quantified equivalence relations."
  :long "<p>It is often useful to introduce equivalence relations such as:</p>

<blockquote>
<i>A === B when for every possible element E, A and B agree on E.</i>
</blockquote>

<p>For some particular notion of what <i>agree</i> means.  This macro gives you
a quick way to introduce such a relation, using @(see defun-sk), and then
automatically prove that it is an equivalence relation.  For instance, an
equivalence such as:</p>

@({
    (defun-sk foo-equiv (a b)
      (forall (x y z)
              (and (bar-equiv (foo a x y)
                              (foo b x y))
                   (baz-equiv (fa a z)
                              (fa b z)))))
})

<p>Can be introduced using:</p>

@({
    (def-universal-equiv foo-equiv (a b)
      :qvars (x y z)
      :equivs ((bar-equiv (foo a x y))
               (baz-equiv (fa a z))))
})

<p>When called with @(':defquant t'), we use @(see defquant) instead of @(see
defun-sk).  This requires that the WITNESS-CP book be included.</p>

<p>Note that @(':qvars') may be omitted, in which case there is no
quantifier (@(see defun-sk)) introduced.  This can be used as a shortcut to
prove that a fixing function induces an equivalence relation, e.g.,</p>

@({
  (def-universal-equiv gfix-equiv
    :equiv-terms ((equal (gfix x))))
 })
<p>produces</p>
@({
 (defun gfix-equiv (x y)
  (equal (gfix x) (gfix y)))

 (defequiv gfix-equiv)
 })

<p>(with appropriate hints to ensure that the @('defequiv') succeeds).</p>")
universal-equiv-equivtermsfunction
(defun universal-equiv-equivterms
  (var1 var2 equivs)
  (if (atom equivs)
    nil
    (let* ((equivname (caar equivs)) (term1 (cadar equivs))
        (term2 (esc-substitute term1 (list (cons var1 var2)))))
      (cons (list equivname term1 term2)
        (universal-equiv-equivterms var1 var2 (cdr equivs))))))
universal-equiv-multi-qvar-bindingsfunction
(defun universal-equiv-multi-qvar-bindings
  (n qvars witnesscall)
  (if (atom qvars)
    nil
    (cons `(,(CAR QVARS) (mv-nth ,N ,WITNESSCALL))
      (universal-equiv-multi-qvar-bindings (1+ n)
        (cdr qvars)
        witnesscall))))
universal-equiv-qvar-bindingsfunction
(defun universal-equiv-qvar-bindings
  (witness var1 var2 qvars)
  (let ((qvars (if (and (consp qvars) (not (consp (cdr qvars))))
         (car qvars)
         qvars)) (term (list witness var1 var2)))
    (if (atom qvars)
      (list (list qvars term))
      (universal-equiv-multi-qvar-bindings 0 qvars term))))
universal-equiv-formfunction
(defun universal-equiv-form
  (equivname qvars
    equivs
    defquant
    witness-dcls
    witness-dcls-p
    parents
    parents-p
    already-definedp
    short
    long
    state)
  (declare (xargs :mode :program))
  (let* ((equivterms `(and . ,(UNIVERSAL-EQUIV-EQUIVTERMS 'X 'Y EQUIVS))) (witness (intern-in-package-of-symbol (concatenate 'string (symbol-name equivname) "-WITNESS")
          equivname))
      (equivname-necc (intern-in-package-of-symbol (concatenate 'string (symbol-name equivname) "-NECC")
          equivname))
      (equivname-refl (intern-in-package-of-symbol (concatenate 'string (symbol-name equivname) "-REFL")
          equivname))
      (equivname-symm (intern-in-package-of-symbol (concatenate 'string (symbol-name equivname) "-SYMM")
          equivname))
      (equivname-trans (intern-in-package-of-symbol (concatenate 'string (symbol-name equivname) "-TRANS")
          equivname))
      (parents (if parents-p
          parents
          (or (get-default-parents (w state)) '(undocumented))))
      (long (or long
          (and parents
            (concatenate 'string
              "<p>This is a universal equivalence, introduced
                     using @(see acl2::def-universal-equiv).</p>"))))
      (long (and long
          (concatenate 'string
            long
            "@(def "
            (full-escape-symbol equivname)
            ")"))))
    `(defsection ,EQUIVNAME
      ,@(AND PARENTS `(:PARENTS ,PARENTS))
      ,@(AND SHORT `(:SHORT ,SHORT))
      ,@(AND LONG `(:LONG ,LONG))
      ,@(AND (NOT ALREADY-DEFINEDP)
       (LIST
        (IF QVARS
            (IF DEFQUANT
                `(DEFQUANT ,EQUIVNAME (X Y) (FORALL ,QVARS ,EQUIVTERMS)
                  ,@(AND WITNESS-DCLS-P `(:WITNESS-DCLS ,WITNESS-DCLS)))
                `(DEFUN-SK ,EQUIVNAME (X Y) ,@WITNESS-DCLS
                           (FORALL ,QVARS ,EQUIVTERMS)))
            `(DEFUN ,EQUIVNAME (X Y) ,EQUIVTERMS))))
      (in-theory (disable ,@(AND QVARS (LIST EQUIVNAME-NECC)) ,EQUIVNAME))
      (local (defthm ,EQUIVNAME-REFL
          (,EQUIVNAME x x)
          :hints (("goal" :in-theory (enable ,EQUIVNAME)))))
      (local (defthm ,EQUIVNAME-SYMM
          (implies (,EQUIVNAME x y) (,EQUIVNAME y x))
          :hints (("goal" ,@(IF QVARS
      `(:USE
        ((:INSTANCE ,EQUIVNAME-NECC
          ,@(UNIVERSAL-EQUIV-QVAR-BINDINGS WITNESS 'Y 'X QVARS)))
        :EXPAND ((,EQUIVNAME Y X)))
      `(:IN-THEORY (ENABLE ,EQUIVNAME)))))))
      (local (defthm ,EQUIVNAME-TRANS
          (implies (and (,EQUIVNAME x y) (,EQUIVNAME y z))
            (,EQUIVNAME x z))
          :hints (("goal" ,@(IF QVARS
      `(:USE
        ((:INSTANCE ,EQUIVNAME-NECC
          ,@(UNIVERSAL-EQUIV-QVAR-BINDINGS WITNESS 'X 'Z QVARS))
         (:INSTANCE ,EQUIVNAME-NECC
          ,@(UNIVERSAL-EQUIV-QVAR-BINDINGS WITNESS 'X 'Z QVARS) (X Y) (Y Z)))
        :EXPAND ((,EQUIVNAME X Z)))
      `(:IN-THEORY (ENABLE ,EQUIVNAME)))))))
      (defequiv ,EQUIVNAME))))
def-universal-equivmacro
(defmacro def-universal-equiv
  (name &key
    qvars
    equiv-terms
    defquant
    (witness-dcls 'nil witness-dcls-p)
    (parents 'nil parents-p)
    already-definedp
    short
    long)
  `(make-event (let ((form (universal-equiv-form ',NAME
           ',QVARS
           ',EQUIV-TERMS
           ',DEFQUANT
           ',WITNESS-DCLS
           ',WITNESS-DCLS-P
           ',PARENTS
           ',PARENTS-P
           ',ALREADY-DEFINEDP
           ',SHORT
           ',LONG
           state)))
      (value form))))