added a mechanism to Nitpick to support custom rendering of terms, and used it for multisets
authorblanchet
Thu, 11 Mar 2010 15:33:45 +0100
changeset 35712 77aa29bf14ee
parent 35711 548d3f16404b
child 35713 428284ee1465
added a mechanism to Nitpick to support custom rendering of terms, and used it for multisets
doc-src/Nitpick/nitpick.tex
src/HOL/Library/Multiset.thy
src/HOL/Nitpick_Examples/Manual_Nits.thy
src/HOL/Tools/Nitpick/HISTORY
src/HOL/Tools/Nitpick/nitpick_model.ML
--- a/doc-src/Nitpick/nitpick.tex	Thu Mar 11 12:22:11 2010 +0100
+++ b/doc-src/Nitpick/nitpick.tex	Thu Mar 11 15:33:45 2010 +0100
@@ -137,8 +137,8 @@
 suggesting several textual improvements.
 % and Perry James for reporting a typo.
 
-\section{Overview}
-\label{overview}
+\section{First Steps}
+\label{first-steps}
 
 This section introduces Nitpick by presenting small examples. If possible, you
 should try out the examples on your workstation. Your theory file should start
@@ -472,7 +472,7 @@
 
 \prew
 \textbf{lemma} ``$\forall n.\; \textit{Suc}~n \mathbin{\not=} n \,\Longrightarrow\, P$'' \\
-\textbf{nitpick} [\textit{card~nat}~= 100, \textit{check\_potential}] \\[2\smallskipamount]
+\textbf{nitpick} [\textit{card~nat}~= 50, \textit{check\_potential}] \\[2\smallskipamount]
 \slshape Warning: The conjecture either trivially holds for the given scopes or lies outside Nitpick's supported
 fragment. Only potential counterexamples may be found. \\[2\smallskipamount]
 Nitpick found a potential counterexample: \\[2\smallskipamount]
@@ -629,7 +629,7 @@
 unlikely that one could be found for smaller cardinalities.
 
 \subsection{Typedefs, Quotient Types, Records, Rationals, and Reals}
-\label{typedefs-records-rationals-and-reals}
+\label{typedefs-quotient-types-records-rationals-and-reals}
 
 Nitpick generally treats types declared using \textbf{typedef} as datatypes
 whose single constructor is the corresponding \textit{Abs\_\kern.1ex} function.
@@ -684,7 +684,26 @@
 
 In the counterexample, $\Abs{(0,\, 0)}$ and $\Abs{(1,\, 0)}$ represent the
 integers $0$ and $1$, respectively. Other representants would have been
-possible---e.g., $\Abs{(5,\, 5)}$ and $\Abs{(12,\, 11)}$.
+possible---e.g., $\Abs{(5,\, 5)}$ and $\Abs{(12,\, 11)}$. If we are going to
+use \textit{my\_int} extensively, it pays off to install a term postprocessor
+that converts the pair notation to the standard mathematical notation:
+
+\prew
+$\textbf{ML}~\,\{{*} \\
+\!\begin{aligned}[t]
+%& ({*}~\,\textit{Proof.context} \rightarrow \textit{string} \rightarrow (\textit{typ} \rightarrow \textit{term~list\/}) \rightarrow \textit{typ} \rightarrow \textit{term} \\[-2pt]
+%& \phantom{(*}~\,{\rightarrow}\;\textit{term}~\,{*}) \\[-2pt]
+& \textbf{fun}\,~\textit{my\_int\_postproc}~\_~\_~\_~T~(\textit{Const}~\_~\$~(\textit{Const}~\_~\$~\textit{t1}~\$~\textit{t2\/})) = {} \\[-2pt]
+& \phantom{fun}\,~\textit{HOLogic.mk\_number}~T~(\textit{snd}~(\textit{HOLogic.dest\_number~t1}) \\[-2pt]
+& \phantom{fun\,~\textit{HOLogic.mk\_number}~T~(}{-}~\textit{snd}~(\textit{HOLogic.dest\_number~t2\/})) \\[-2pt]
+& \phantom{fun}\!{\mid}\,~\textit{my\_int\_postproc}~\_~\_~\_~\_~t = t \\[-2pt]
+{*}\}\end{aligned}$ \\[2\smallskipamount]
+$\textbf{setup}~\,\{{*} \\
+\!\begin{aligned}[t]
+& \textit{Nitpick.register\_term\_postprocessor}~\!\begin{aligned}[t]
+  & @\{\textrm{typ}~\textit{my\_int}\}~\textit{my\_int\_postproc}\end{aligned} \\[-2pt]
+{*}\}\end{aligned}$
+\postw
 
 Records are also handled as datatypes with a single constructor:
 
@@ -771,25 +790,25 @@
 
 \prew
 \textbf{lemma} ``$\exists n.\; \textit{even}~n \mathrel{\land} \textit{even}~(\textit{Suc}~n)$'' \\
-\textbf{nitpick}~[\textit{card nat}~= 100, \textit{unary\_ints}, \textit{verbose}] \\[2\smallskipamount]
+\textbf{nitpick}~[\textit{card nat}~= 50, \textit{unary\_ints}, \textit{verbose}] \\[2\smallskipamount]
 \slshape The inductive predicate ``\textit{even}'' was proved well-founded.
 Nitpick can compute it efficiently. \\[2\smallskipamount]
 Trying 1 scope: \\
-\hbox{}\qquad \textit{card nat}~= 100. \\[2\smallskipamount]
-Nitpick found a potential counterexample for \textit{card nat}~= 100: \\[2\smallskipamount]
+\hbox{}\qquad \textit{card nat}~= 50. \\[2\smallskipamount]
+Nitpick found a potential counterexample for \textit{card nat}~= 50: \\[2\smallskipamount]
 \hbox{}\qquad Empty assignment \\[2\smallskipamount]
 Nitpick could not find a better counterexample. \\[2\smallskipamount]
 Total time: 2274 ms.
 \postw
 
 No genuine counterexample is possible because Nitpick cannot rule out the
-existence of a natural number $n \ge 100$ such that both $\textit{even}~n$ and
+existence of a natural number $n \ge 50$ such that both $\textit{even}~n$ and
 $\textit{even}~(\textit{Suc}~n)$ are true. To help Nitpick, we can bound the
 existential quantifier:
 
 \prew
-\textbf{lemma} ``$\exists n \mathbin{\le} 99.\; \textit{even}~n \mathrel{\land} \textit{even}~(\textit{Suc}~n)$'' \\
-\textbf{nitpick}~[\textit{card nat}~= 100, \textit{unary\_ints}] \\[2\smallskipamount]
+\textbf{lemma} ``$\exists n \mathbin{\le} 49.\; \textit{even}~n \mathrel{\land} \textit{even}~(\textit{Suc}~n)$'' \\
+\textbf{nitpick}~[\textit{card nat}~= 50, \textit{unary\_ints}] \\[2\smallskipamount]
 \slshape Nitpick found a counterexample: \\[2\smallskipamount]
 \hbox{}\qquad Empty assignment
 \postw
@@ -1211,26 +1230,26 @@
 Trying 8 scopes: \\
 \hbox{}\qquad \textit{card} $'a$~= 1, \textit{card} $'b$~= 1,
 \textit{card} \textit{nat}~= 1, \textit{card} ``$('a \times {'}b)$
-\textit{list}''~= 1, \\
-\hbox{}\qquad\quad \textit{card} ``\kern1pt$'a$ \textit{list}''~= 1, and
-\textit{card} ``\kern1pt$'b$ \textit{list}''~= 1. \\
+\textit{list\/}''~= 1, \\
+\hbox{}\qquad\quad \textit{card} ``\kern1pt$'a$ \textit{list\/}''~= 1, and
+\textit{card} ``\kern1pt$'b$ \textit{list\/}''~= 1. \\
 \hbox{}\qquad \textit{card} $'a$~= 2, \textit{card} $'b$~= 2,
 \textit{card} \textit{nat}~= 2, \textit{card} ``$('a \times {'}b)$
-\textit{list}''~= 2, \\
-\hbox{}\qquad\quad \textit{card} ``\kern1pt$'a$ \textit{list}''~= 2, and
-\textit{card} ``\kern1pt$'b$ \textit{list}''~= 2. \\
+\textit{list\/}''~= 2, \\
+\hbox{}\qquad\quad \textit{card} ``\kern1pt$'a$ \textit{list\/}''~= 2, and
+\textit{card} ``\kern1pt$'b$ \textit{list\/}''~= 2. \\
 \hbox{}\qquad $\qquad\vdots$ \\[.5\smallskipamount]
 \hbox{}\qquad \textit{card} $'a$~= 8, \textit{card} $'b$~= 8,
 \textit{card} \textit{nat}~= 8, \textit{card} ``$('a \times {'}b)$
-\textit{list}''~= 8, \\
-\hbox{}\qquad\quad \textit{card} ``\kern1pt$'a$ \textit{list}''~= 8, and
-\textit{card} ``\kern1pt$'b$ \textit{list}''~= 8.
+\textit{list\/}''~= 8, \\
+\hbox{}\qquad\quad \textit{card} ``\kern1pt$'a$ \textit{list\/}''~= 8, and
+\textit{card} ``\kern1pt$'b$ \textit{list\/}''~= 8.
 \\[2\smallskipamount]
 Nitpick found a counterexample for
 \textit{card} $'a$~= 5, \textit{card} $'b$~= 5,
 \textit{card} \textit{nat}~= 5, \textit{card} ``$('a \times {'}b)$
-\textit{list}''~= 5, \textit{card} ``\kern1pt$'a$ \textit{list}''~= 5, and
-\textit{card} ``\kern1pt$'b$ \textit{list}''~= 5:
+\textit{list\/}''~= 5, \textit{card} ``\kern1pt$'a$ \textit{list\/}''~= 5, and
+\textit{card} ``\kern1pt$'b$ \textit{list\/}''~= 5:
 \\[2\smallskipamount]
 \hbox{}\qquad Free variables: \nopagebreak \\
 \hbox{}\qquad\qquad $\textit{xs} = [a_1, a_2]$ \\
@@ -1851,7 +1870,7 @@
 The number of options can be overwhelming at first glance. Do not let that worry
 you: Nitpick's defaults have been chosen so that it almost always does the right
 thing, and the most important options have been covered in context in
-\S\ref{overview}.
+\S\ref{first-steps}.
 
 The descriptions below refer to the following syntactic quantities:
 
@@ -2622,7 +2641,7 @@
 
 then Nitpick assumes that the definition was made using an inductive package and
 based on the introduction rules marked with \textit{nitpick\_\allowbreak
-ind\_\allowbreak intros} tries to determine whether the definition is
+\allowbreak intros} tries to determine whether the definition is
 well-founded.
 \end{enum}
 \end{enum}
@@ -2664,7 +2683,8 @@
 Nitpick provides a rich Standard ML interface used mainly for internal purposes
 and debugging. Among the most interesting functions exported by Nitpick are
 those that let you invoke the tool programmatically and those that let you
-register and unregister custom coinductive datatypes.
+register and unregister custom coinductive datatypes as well as term
+postprocessors.
 
 \subsection{Invocation of Nitpick}
 \label{invocation-of-nitpick}
@@ -2695,7 +2715,7 @@
 put into a \textit{params} record. Here is an example:
 
 \prew
-$\textbf{val}\,~\textit{params} = \textit{Nitpick\_Isar.default\_params}~\textit{thy}~[(\textrm{``}\textrm{timeout}\textrm{''},\, \textrm{``}\textrm{none}\textrm{''})]$ \\
+$\textbf{val}\,~\textit{params} = \textit{Nitpick\_Isar.default\_params}~\textit{thy}~[(\textrm{``}\textrm{timeout\/}\textrm{''},\, \textrm{``}\textrm{none}\textrm{''})]$ \\
 $\textbf{val}\,~(\textit{outcome},\, \textit{state}') = \textit{Nitpick.pick\_nits\_in\_subgoal}~\begin{aligned}[t]
 & \textit{state}~\textit{params}~\textit{false} \\[-2pt]
 & \textit{subgoal}\end{aligned}$
@@ -2726,7 +2746,7 @@
 \prew
 $\textbf{setup}~\,\{{*}\,~\!\begin{aligned}[t]
 & \textit{Nitpick.register\_codatatype} \\[-2pt]
-& \qquad @\{\antiq{typ}~``\kern1pt'a~\textit{llist}\textrm{''}\}~@\{\antiq{const\_name}~ \textit{llist\_case}\} \\[-2pt] %% TYPESETTING
+& \qquad @\{\antiq{typ}~``\kern1pt'a~\textit{llist\/}\textrm{''}\}~@\{\antiq{const\_name}~ \textit{llist\_case}\} \\[-2pt] %% TYPESETTING
 & \qquad (\textit{map}~\textit{dest\_Const}~[@\{\antiq{term}~\textit{LNil}\},\, @\{\antiq{term}~\textit{LCons}\}])\,\ {*}\}\end{aligned}$
 \postw
 
@@ -2740,7 +2760,7 @@
 
 \prew
 $\textbf{setup}~\,\{{*}\,~\textit{Nitpick.unregister\_codatatype}~@\{\antiq{typ}~``
-\kern1pt'a~\textit{list}\textrm{''}\}\ \,{*}\}$
+\kern1pt'a~\textit{list\/}\textrm{''}\}\ \,{*}\}$
 \postw
 
 Inductive datatypes can be registered as coinductive datatypes, given
@@ -2748,6 +2768,26 @@
 the use of the inductive constructors---Nitpick will generate an error if they
 are needed.
 
+\subsection{Registration of Term Postprocessors}
+\label{registration-of-term-postprocessors}
+
+It is possible to change the output of any term that Nitpick considers a
+datatype by registering a term postprocessor. The interface for registering and
+unregistering postprocessors consists of the following pair of functions defined
+in the \textit{Nitpick} structure:
+
+\prew
+$\textbf{type}\,~\textit{term\_postprocessor}\,~{=} {}$ \\
+$\hbox{}\quad\textit{Proof.context} \rightarrow \textit{string} \rightarrow (\textit{typ} \rightarrow \textit{term~list\/}) \rightarrow \textit{typ} \rightarrow \textit{term} \rightarrow \textit{term}$ \\
+$\textbf{val}\,~\textit{register\_term\_postprocessors} : {}$ \\
+$\hbox{}\quad\textit{typ} \rightarrow \textit{term\_postprocessor} \rightarrow \textit{theory} \rightarrow \textit{theory}$ \\
+$\textbf{val}\,~\textit{unregister\_term\_postprocessors} :\,
+\textit{typ} \rightarrow \textit{theory} \rightarrow \textit{theory}$
+\postw
+
+\S\ref{typedefs-quotient-types-records-rationals-and-reals} and
+\texttt{src/HOL/Library/Multiset.thy} illustrate this feature in context.
+
 \section{Known Bugs and Limitations}
 \label{known-bugs-and-limitations}
 
--- a/src/HOL/Library/Multiset.thy	Thu Mar 11 12:22:11 2010 +0100
+++ b/src/HOL/Library/Multiset.thy	Thu Mar 11 15:33:45 2010 +0100
@@ -1729,4 +1729,33 @@
   "M \<subset># N ==> (\<not> P ==> N \<subset># (M::'a::order multiset)) ==> P"
   by (fact multiset_order.less_asym)
 
+ML {*
+(* Proof.context -> string -> (typ -> term list) -> typ -> term -> term *)
+fun multiset_postproc _ maybe_name all_values (T as Type (_, [elem_T]))
+                      (Const _ $ t') =
+    let
+      val (maybe_opt, ps) =
+        Nitpick_Model.dest_plain_fun t' ||> op ~~
+        ||> map (apsnd (snd o HOLogic.dest_number))
+      fun elems_for t =
+        case AList.lookup (op =) ps t of
+          SOME n => replicate n t
+        | NONE => [Const (maybe_name, elem_T --> elem_T) $ t]
+    in
+      case maps elems_for (all_values elem_T) @
+           (if maybe_opt then [Const (Nitpick_Model.unrep, elem_T)] else []) of
+        [] => Const (@{const_name zero_class.zero}, T)
+      | ts => foldl1 (fn (t1, t2) =>
+                         Const (@{const_name plus_class.plus}, T --> T --> T)
+                         $ t1 $ t2)
+                     (map (curry (op $) (Const (@{const_name single},
+                                                elem_T --> T))) ts)
+    end
+  | multiset_postproc _ _ _ _ t = t
+*}
+
+setup {*
+Nitpick.register_term_postprocessor @{typ "'a multiset"} multiset_postproc
+*}
+
 end
\ No newline at end of file
--- a/src/HOL/Nitpick_Examples/Manual_Nits.thy	Thu Mar 11 12:22:11 2010 +0100
+++ b/src/HOL/Nitpick_Examples/Manual_Nits.thy	Thu Mar 11 15:33:45 2010 +0100
@@ -118,10 +118,11 @@
 oops
 
 ML {*
-(* Proof.context -> typ -> term -> term *)
-fun my_int_postproc _ T (Const _ $ (Const _ $ t1 $ t2)) =
-    HOLogic.mk_number T (snd (HOLogic.dest_number t1) - snd (HOLogic.dest_number t2))
-  | my_int_postproc _ _ t = t
+(* Proof.context -> string -> (typ -> term list) -> typ -> term -> term *)
+fun my_int_postproc _ _ _ T (Const _ $ (Const _ $ t1 $ t2)) =
+    HOLogic.mk_number T (snd (HOLogic.dest_number t1)
+                         - snd (HOLogic.dest_number t2))
+  | my_int_postproc _ _ _ _ t = t
 *}
 
 setup {* Nitpick.register_term_postprocessor @{typ my_int} my_int_postproc *}
--- a/src/HOL/Tools/Nitpick/HISTORY	Thu Mar 11 12:22:11 2010 +0100
+++ b/src/HOL/Tools/Nitpick/HISTORY	Thu Mar 11 15:33:45 2010 +0100
@@ -7,6 +7,7 @@
   * Added support for quotient types
   * Added support for local definitions (for "function" and "termination"
     proofs)
+  * Added support for term postprocessors
   * Optimized "Multiset.multiset" and "FinFun.finfun"
   * Improved efficiency of "destroy_constrs" optimization
   * Fixed soundness bugs related to "destroy_constrs" optimization and record
--- a/src/HOL/Tools/Nitpick/nitpick_model.ML	Thu Mar 11 12:22:11 2010 +0100
+++ b/src/HOL/Tools/Nitpick/nitpick_model.ML	Thu Mar 11 15:33:45 2010 +0100
@@ -16,15 +16,20 @@
     show_skolems: bool,
     show_datatypes: bool,
     show_consts: bool}
-  type term_postprocessor = Proof.context -> typ -> term -> term
+  type term_postprocessor =
+    Proof.context -> string -> (typ -> term list) -> typ -> term -> term
 
   structure NameTable : TABLE
 
+  val irrelevant : string
+  val unknown : string
+  val unrep : string
   val register_term_postprocessor :
     typ -> term_postprocessor -> theory -> theory
   val unregister_term_postprocessor : typ -> theory -> theory
   val tuple_list_for_name :
     nut NameTable.table -> Kodkod.raw_bound list -> nut -> int list list
+  val dest_plain_fun : term -> bool * (term list * term list)
   val reconstruct_hol_model :
     params -> scope -> (term option * int list) list -> styp list -> nut list
     -> nut list -> nut list -> nut NameTable.table -> Kodkod.raw_bound list
@@ -51,7 +56,8 @@
   show_datatypes: bool,
   show_consts: bool}
 
-type term_postprocessor = Proof.context -> typ -> term -> term
+type term_postprocessor =
+  Proof.context -> string -> (typ -> term list) -> typ -> term -> term
 
 structure Data = Theory_Data(
   type T = (typ * term_postprocessor) list
@@ -59,6 +65,7 @@
   val extend = I
   fun merge (ps1, ps2) = AList.merge (op =) (K true) (ps1, ps2))
 
+val irrelevant = "_"
 val unknown = "?"
 val unrep = "\<dots>"
 val maybe_mixfix = "_\<^sup>?"
@@ -182,9 +189,9 @@
     (* typ -> typ -> (term * term) list -> term *)
     fun aux T1 T2 [] =
         Const (if maybe_opt then opt_flag else non_opt_flag, T1 --> T2)
-      | aux T1 T2 ((t1, t2) :: ps) =
+      | aux T1 T2 ((t1, t2) :: tps) =
         Const (@{const_name fun_upd}, (T1 --> T2) --> T1 --> T2 --> T1 --> T2)
-        $ aux T1 T2 ps $ t1 $ t2
+        $ aux T1 T2 tps $ t1 $ t2
   in aux T1 T2 o rev end
 (* term -> bool *)
 fun is_plain_fun (Const (s, _)) = (s = opt_flag orelse s = non_opt_flag)
@@ -195,9 +202,12 @@
 val dest_plain_fun =
   let
     (* term -> bool * (term list * term list) *)
-    fun aux (Const (s, _)) = (s <> non_opt_flag, ([], []))
+    fun aux (Abs (_, _, Const (s, _))) = (s <> irrelevant, ([], []))
+      | aux (Const (s, _)) = (s <> non_opt_flag, ([], []))
       | aux (Const (@{const_name fun_upd}, _) $ t0 $ t1 $ t2) =
-        let val (s, (ts1, ts2)) = aux t0 in (s, (t1 :: ts1, t2 :: ts2)) end
+        let val (maybe_opt, (ts1, ts2)) = aux t0 in
+          (maybe_opt, (t1 :: ts1, t2 :: ts2))
+        end
       | aux t = raise TERM ("Nitpick_Model.dest_plain_fun", [t])
   in apsnd (pairself rev) o aux end
 
@@ -227,22 +237,22 @@
       (* typ -> typ -> typ -> typ -> term -> term *)
       fun do_curry T1 T1a T1b T2 t =
         let
-          val (maybe_opt, ps) = dest_plain_fun t
-          val ps =
-            ps |>> map (break_in_two T1 T1a T1b)
-               |> uncurry (map2 (fn (t1a, t1b) => fn t2 => (t1a, (t1b, t2))))
-               |> AList.coalesce (op =)
-               |> map (apsnd (make_plain_fun maybe_opt T1b T2))
-        in make_plain_fun maybe_opt T1a (T1b --> T2) ps end
+          val (maybe_opt, tsp) = dest_plain_fun t
+          val tps =
+            tsp |>> map (break_in_two T1 T1a T1b)
+                |> uncurry (map2 (fn (t1a, t1b) => fn t2 => (t1a, (t1b, t2))))
+                |> AList.coalesce (op =)
+                |> map (apsnd (make_plain_fun maybe_opt T1b T2))
+        in make_plain_fun maybe_opt T1a (T1b --> T2) tps end
       (* typ -> typ -> term -> term *)
       and do_uncurry T1 T2 t =
         let
           val (maybe_opt, tsp) = dest_plain_fun t
-          val ps =
+          val tps =
             tsp |> op ~~
                 |> maps (fn (t1, t2) =>
                             multi_pair_up T1 t1 (snd (dest_plain_fun t2)))
-        in make_plain_fun maybe_opt T1 T2 ps end
+        in make_plain_fun maybe_opt T1 T2 tps end
       (* typ -> typ -> typ -> typ -> term -> term *)
       and do_arrow T1' T2' _ _ (Const (s, _)) = Const (s, T1' --> T2')
         | do_arrow T1' T2' T1 T2
@@ -291,22 +301,37 @@
 (* theory -> typ * typ -> bool *)
 fun varified_type_match thy (candid_T, pat_T) =
   strict_type_match thy (candid_T, Logic.varifyT pat_T)
-(* Proof.context -> typ -> term -> term *)
-fun postprocess_term ctxt T =
-  let val thy = ProofContext.theory_of ctxt in
-    if null (Data.get thy) then
-      I
-    else
-      (T |> AList.lookup (varified_type_match thy) (Data.get thy)
-         |> the_default (K (K I))) ctxt T
-  end
 
+(* atom_pool -> (string * string) * (string * string) -> scope -> nut list
+   -> nut list -> nut list -> nut NameTable.table -> KK.raw_bound list -> typ
+   -> term list *)
+fun all_values_of_type pool wacky_names (scope as {card_assigns, ...} : scope)
+                       sel_names rel_table bounds card T =
+  let
+    val card = if card = 0 then card_of_type card_assigns T else card
+    (* nat -> term *)
+    fun nth_value_of_type n =
+      let
+        (* bool -> term *)
+        fun term unfold =
+          reconstruct_term unfold pool wacky_names scope sel_names rel_table
+                           bounds T T (Atom (card, 0)) [[n]]
+      in
+        case term false of
+          t as Const (s, _) =>
+          if String.isPrefix cyclic_const_prefix s then
+            HOLogic.mk_eq (t, term true)
+          else
+            t
+        | t => t
+      end
+  in index_seq 0 card |> map nth_value_of_type |> sort nice_term_ord end
 (* bool -> atom_pool -> (string * string) * (string * string) -> scope
    -> nut list -> nut list -> nut list -> nut NameTable.table
    -> KK.raw_bound list -> typ -> typ -> rep -> int list list -> term *)
-fun reconstruct_term unfold pool ((maybe_name, abs_name), _)
-        ({hol_ctxt as {ctxt, thy, stds, ...}, binarize, card_assigns, bits,
-          datatypes, ofs, ...} : scope) sel_names rel_table bounds =
+and reconstruct_term unfold pool (wacky_names as ((maybe_name, abs_name), _))
+        (scope as {hol_ctxt as {ctxt, thy, stds, ...}, binarize, card_assigns,
+                   bits, datatypes, ofs, ...}) sel_names rel_table bounds =
   let
     val for_auto = (maybe_name = "")
     (* int list list -> int *)
@@ -318,6 +343,25 @@
         fold (fn j => Integer.add (reasonable_power 2 j |> j = bits ? op ~))
              js 0
       end
+    (* typ -> term list *)
+    val all_values =
+      all_values_of_type pool wacky_names scope sel_names rel_table bounds 0
+    (* typ -> term -> term *)
+    fun postprocess_term (Type (@{type_name fun}, _)) = I
+      | postprocess_term T =
+        if null (Data.get thy) then
+          I
+        else case AList.lookup (varified_type_match thy) (Data.get thy) T of
+          SOME postproc => postproc ctxt maybe_name all_values T
+        | NONE => I
+    (* typ list -> term -> term *)
+    fun postprocess_subterms Ts (t1 $ t2) =
+        let val t = postprocess_subterms Ts t1 $ postprocess_subterms Ts t2 in
+          postprocess_term (fastype_of1 (Ts, t)) t
+        end
+      | postprocess_subterms Ts (Abs (s, T, t')) =
+        Abs (s, T, postprocess_subterms (T :: Ts) t')
+      | postprocess_subterms Ts t = postprocess_term (fastype_of1 (Ts, t)) t
     (* bool -> typ -> typ -> (term * term) list -> term *)
     fun make_set maybe_opt T1 T2 tps =
       let
@@ -352,16 +396,16 @@
                                   (T1 --> T2) --> T1 --> T2 --> T1 --> T2)
         (* (term * term) list -> term *)
         fun aux' [] = Const (@{const_abbrev Map.empty}, T1 --> T2)
-          | aux' ((t1, t2) :: ps) =
+          | aux' ((t1, t2) :: tps) =
             (case t2 of
-               Const (@{const_name None}, _) => aux' ps
-             | _ => update_const $ aux' ps $ t1 $ t2)
-        fun aux ps =
+               Const (@{const_name None}, _) => aux' tps
+             | _ => update_const $ aux' tps $ t1 $ t2)
+        fun aux tps =
           if maybe_opt andalso not (is_complete_type datatypes false T1) then
-            update_const $ aux' ps $ Const (unrep, T1)
+            update_const $ aux' tps $ Const (unrep, T1)
             $ (Const (@{const_name Some}, T2' --> T2) $ Const (unknown, T2'))
           else
-            aux' ps
+            aux' tps
       in aux end
     (* typ list -> term -> term *)
     fun polish_funs Ts t =
@@ -396,7 +440,11 @@
              | Abs (s, T, t') => Abs (s, T, polish_funs (T :: Ts) t')
              | Const (s, Type (@{type_name fun}, [T1, T2])) =>
                if s = opt_flag orelse s = non_opt_flag then
-                 Abs ("x", T1, Const (unknown, T2))
+                 Abs ("x", T1,
+                      Const (if is_complete_type datatypes false T1 then
+                               irrelevant
+                             else
+                               unknown, T2))
                else
                  t
              | t => t
@@ -541,7 +589,6 @@
                   t
               end
           end
-          |> postprocess_term ctxt T
     (* (typ * int) list -> int -> rep -> typ -> typ -> typ -> int list
        -> term *)
     and term_for_vect seen k R T1 T2 T' js =
@@ -594,7 +641,10 @@
         raise ARG ("Nitpick_Model.reconstruct_term.term_for_rep",
                    Refute.string_of_typ T ^ " " ^ string_for_rep R ^ " " ^
                    string_of_int (length jss))
-  in polish_funs [] o unarize_unbox_etc_term oooo term_for_rep true [] end
+  in
+    postprocess_subterms [] o polish_funs [] o unarize_unbox_etc_term
+    oooo term_for_rep true []
+  end
 
 (* atom_pool -> scope -> nut list -> nut NameTable.table -> KK.raw_bound list
    -> nut -> term *)
@@ -702,14 +752,14 @@
             t
         | t => t
       end
-    (* nat -> typ -> typ list *)
-    fun all_values_of_type card T =
-      index_seq 0 card |> map (nth_value_of_type card T) |> sort nice_term_ord
+    (* nat -> typ -> term list *)
+    val all_values =
+      all_values_of_type pool wacky_names scope sel_names rel_table bounds
     (* dtype_spec list -> dtype_spec -> bool *)
     fun is_codatatype_wellformed (cos : dtype_spec list)
                                  ({typ, card, ...} : dtype_spec) =
       let
-        val ts = all_values_of_type card typ
+        val ts = all_values card typ
         val max_depth = Integer.sum (map #card cos)
       in
         forall (not o bisimilar_values (map #typ cos) max_depth)
@@ -750,7 +800,7 @@
             | _ => []) @
            [Pretty.str "=",
             Pretty.enum "," "{" "}"
-                (map (Syntax.pretty_term ctxt) (all_values_of_type card typ) @
+                (map (Syntax.pretty_term ctxt) (all_values card typ) @
                  (if fun_from_pair complete false then []
                   else [Pretty.str unrep]))]))
     (* typ -> dtype_spec list *)