fixed bug in the "fast_descrs" optimization;
authorblanchet
Tue, 14 Sep 2010 12:52:50 +0200
changeset 39358 6bc2f7971df0
parent 39357 fe84bc307be6
child 39359 6f49c7fbb1b1
fixed bug in the "fast_descrs" optimization; the bug is that two sets may actually be the same but because of the three-valued logic a different "The" or "Eps" is chosen; e.g. consider the set {1, 2}. If it is approximated in one place as {1, 2?} and in another place as {1?, 2}, then "Eps" would return 1 in the first case and 2 in the second case. This is of course wrong, because both sets potentially represent {1, 2}. The current fix has a very negative impact on precision.
src/HOL/Tools/Nitpick/nitpick_kodkod.ML
--- a/src/HOL/Tools/Nitpick/nitpick_kodkod.ML	Tue Sep 14 11:18:40 2010 +0200
+++ b/src/HOL/Tools/Nitpick/nitpick_kodkod.ML	Tue Sep 14 12:52:50 2010 +0200
@@ -1029,19 +1029,20 @@
     val bool_atom_R = Atom (2, main_j0)
     val false_atom = KK.Atom bool_j0
     val true_atom = KK.Atom (bool_j0 + 1)
-
     fun formula_from_opt_atom polar j0 r =
       case polar of
         Neg => kk_not (kk_rel_eq r (KK.Atom j0))
       | _ => kk_rel_eq r (KK.Atom (j0 + 1))
     val formula_from_atom = formula_from_opt_atom Pos
-
     val unpack_formulas =
       map (formula_from_atom bool_j0) oo unpack_vect_in_chunks kk 1
     fun kk_vect_set_bool_op connective k r1 r2 =
       fold1 kk_and (map2 connective (unpack_formulas k r1)
                          (unpack_formulas k r2))
-
+    fun is_surely_singleton R r =
+      kk_and (kk_subset (full_rel_for_rep R)
+                        (kk_join r (full_rel_for_rep bool_atom_R)))
+             (kk_one (kk_join r true_atom))
     fun to_f u =
       case rep_of u of
         Formula polar =>
@@ -1475,7 +1476,7 @@
       | Op2 (The, _, R, u1, u2) =>
         if is_opt_rep R then
           let val r1 = to_opt (Func (unopt_rep R, bool_atom_R)) u1 in
-            kk_rel_if (kk_one (kk_join r1 true_atom)) (kk_join r1 true_atom)
+            kk_rel_if (is_surely_singleton R r1) (kk_join r1 true_atom)
                       (kk_rel_if (kk_or (kk_some (kk_join r1 true_atom))
                                         (kk_subset (full_rel_for_rep R)
                                                    (kk_join r1 false_atom)))
@@ -1491,7 +1492,7 @@
             val r1 = to_rep (Func (unopt_rep R, Opt bool_atom_R)) u1
             val r2 = to_rep R u2
           in
-            kk_union (kk_rel_if (kk_one (kk_join r1 true_atom))
+            kk_union (kk_rel_if (is_surely_singleton R r1)
                                 (kk_join r1 true_atom) (empty_rel_for_rep R))
                      (kk_rel_if (kk_or (kk_subset r2 (kk_join r1 true_atom))
                                        (kk_subset (full_rel_for_rep R)