src/HOL/Tools/Qelim/presburger.ML
author wenzelm
Tue, 03 Jul 2007 17:17:04 +0200
changeset 23530 438c5d2db482
parent 23499 7e27947d92d7
child 23880 64b9806e160b
permissions -rw-r--r--
CONVERSION tactical;
wenzelm@23466
     1
(*  Title:      HOL/Tools/Presburger/presburger.ML
wenzelm@23466
     2
    ID:         $Id$
wenzelm@23466
     3
    Author:     Amine Chaieb, TU Muenchen
wenzelm@23466
     4
*)
wenzelm@23466
     5
wenzelm@23466
     6
signature PRESBURGER =
wenzelm@23499
     7
sig
wenzelm@23499
     8
  val cooper_tac: bool -> thm list -> thm list -> Proof.context -> int -> tactic
wenzelm@23466
     9
end;
wenzelm@23466
    10
wenzelm@23466
    11
structure Presburger : PRESBURGER = 
wenzelm@23466
    12
struct
wenzelm@23466
    13
wenzelm@23466
    14
open Conv;
wenzelm@23466
    15
val comp_ss = HOL_ss addsimps @{thms "Groebner_Basis.comp_arith"};
wenzelm@23466
    16
wenzelm@23499
    17
fun strip_objimp ct =
wenzelm@23499
    18
  (case Thm.term_of ct of
wenzelm@23499
    19
    Const ("op -->", _) $ _ $ _ =>
wenzelm@23499
    20
      let val (A, B) = Thm.dest_binop ct
wenzelm@23499
    21
      in A :: strip_objimp B end
wenzelm@23499
    22
  | _ => [ct]);
wenzelm@23466
    23
wenzelm@23466
    24
fun strip_objall ct = 
wenzelm@23466
    25
 case term_of ct of 
wenzelm@23466
    26
  Const ("All", _) $ Abs (xn,xT,p) => 
wenzelm@23466
    27
   let val (a,(v,t')) = (apsnd (Thm.dest_abs (SOME xn)) o Thm.dest_comb) ct
wenzelm@23466
    28
   in apfst (cons (a,v)) (strip_objall t')
wenzelm@23466
    29
   end
wenzelm@23466
    30
| _ => ([],ct);
wenzelm@23466
    31
wenzelm@23466
    32
local
wenzelm@23466
    33
  val all_maxscope_ss = 
wenzelm@23466
    34
     HOL_basic_ss addsimps map (fn th => th RS sym) @{thms "all_simps"}
wenzelm@23466
    35
in
wenzelm@23499
    36
fun thin_prems_tac P = simp_tac all_maxscope_ss THEN'
wenzelm@23499
    37
  CSUBGOAL (fn (p', i) =>
wenzelm@23466
    38
    let
wenzelm@23466
    39
     val (qvs, p) = strip_objall (Thm.dest_arg p')
wenzelm@23466
    40
     val (ps, c) = split_last (strip_objimp p)
wenzelm@23466
    41
     val qs = filter P ps
wenzelm@23466
    42
     val q = if P c then c else @{cterm "False"}
wenzelm@23466
    43
     val ng = fold_rev (fn (a,v) => fn t => Thm.capply a (Thm.cabs v t)) qvs 
wenzelm@23466
    44
         (fold_rev (fn p => fn q => Thm.capply (Thm.capply @{cterm "op -->"} p) q) qs q)
wenzelm@23466
    45
     val g = Thm.capply (Thm.capply @{cterm "op ==>"} (Thm.capply @{cterm "Trueprop"} ng)) p'
wenzelm@23466
    46
     val ntac = (case qs of [] => q aconvc @{cterm "False"}
wenzelm@23466
    47
                         | _ => false)
wenzelm@23466
    48
    in 
wenzelm@23499
    49
    if ntac then no_tac
wenzelm@23499
    50
      else rtac (Goal.prove_internal [] g (K (blast_tac HOL_cs 1))) i
wenzelm@23466
    51
    end)
wenzelm@23466
    52
end;
wenzelm@23466
    53
wenzelm@23466
    54
local
wenzelm@23466
    55
 fun ty cts t = 
wenzelm@23466
    56
 if not (typ_of (ctyp_of_term t) mem [HOLogic.intT, HOLogic.natT]) then false 
wenzelm@23466
    57
    else case term_of t of 
wenzelm@23466
    58
      c$_$_ => not (member (op aconv) cts c)
wenzelm@23466
    59
    | c$_ => not (member (op aconv) cts c)
wenzelm@23466
    60
    | c => not (member (op aconv) cts c)
wenzelm@23466
    61
wenzelm@23466
    62
 val term_constants =
wenzelm@23466
    63
  let fun h acc t = case t of
wenzelm@23466
    64
    Const _ => insert (op aconv) t acc
wenzelm@23466
    65
  | a$b => h (h acc a) b
wenzelm@23466
    66
  | Abs (_,_,t) => h acc t
wenzelm@23466
    67
  | _ => acc
wenzelm@23466
    68
 in h [] end;
wenzelm@23466
    69
in 
wenzelm@23466
    70
fun is_relevant ctxt ct = 
wenzelm@23466
    71
  gen_subset (op aconv) (term_constants (term_of ct) , snd (CooperData.get ctxt))
wenzelm@23466
    72
 andalso forall (fn Free (_,T) => T = HOLogic.intT) (term_frees (term_of ct))
wenzelm@23466
    73
 andalso forall (fn Var (_,T) => T = HOLogic.intT) (term_vars (term_of ct));
wenzelm@23466
    74
wenzelm@23466
    75
fun int_nat_terms ctxt ct =
wenzelm@23466
    76
 let 
wenzelm@23466
    77
  val cts = snd (CooperData.get ctxt)
wenzelm@23466
    78
  fun h acc t = if ty cts t then insert (op aconvc) t acc else
wenzelm@23466
    79
   case (term_of t) of
wenzelm@23466
    80
    _$_ => h (h acc (Thm.dest_arg t)) (Thm.dest_fun t)
wenzelm@23466
    81
  | Abs(_,_,_) => Thm.dest_abs NONE t ||> h acc |> uncurry (remove (op aconvc))
wenzelm@23466
    82
  | _ => acc
wenzelm@23466
    83
 in h [] ct end
wenzelm@23466
    84
end;
wenzelm@23466
    85
wenzelm@23499
    86
fun generalize_tac f = CSUBGOAL (fn (p, i) => PRIMITIVE (fn st =>
wenzelm@23499
    87
 let 
wenzelm@23466
    88
   fun all T = Drule.cterm_rule (instantiate' [SOME T] []) @{cpat "all"}
wenzelm@23466
    89
   fun gen x t = Thm.capply (all (ctyp_of_term x)) (Thm.cabs x t)
wenzelm@23466
    90
   val ts = sort (fn (a,b) => Term.fast_term_ord (term_of a, term_of b)) (f p)
wenzelm@23466
    91
   val p' = fold_rev gen ts p
wenzelm@23499
    92
 in implies_intr p' (implies_elim st (fold forall_elim ts (assume p'))) end));
wenzelm@23466
    93
wenzelm@23466
    94
local
wenzelm@23466
    95
val ss1 = comp_ss
wenzelm@23466
    96
  addsimps simp_thms @ [@{thm "nat_number_of_def"}, @{thm "zdvd_int"}] 
wenzelm@23466
    97
      @ map (fn r => r RS sym) 
wenzelm@23466
    98
        [@{thm "int_int_eq"}, @{thm "zle_int"}, @{thm "zless_int"}, @{thm "zadd_int"}, 
wenzelm@23466
    99
         @{thm "zmult_int"}]
wenzelm@23466
   100
    addsplits [@{thm "zdiff_int_split"}]
wenzelm@23466
   101
wenzelm@23466
   102
val ss2 = HOL_basic_ss
wenzelm@23466
   103
  addsimps [@{thm "nat_0_le"}, @{thm "int_nat_number_of"},
wenzelm@23466
   104
            @{thm "all_nat"}, @{thm "ex_nat"}, @{thm "number_of1"}, 
wenzelm@23466
   105
            @{thm "number_of2"}, @{thm "int_0"}, @{thm "int_1"}, @{thm "Suc_plus1"}]
wenzelm@23466
   106
  addcongs [@{thm "conj_le_cong"}, @{thm "imp_le_cong"}]
wenzelm@23466
   107
val div_mod_ss = HOL_basic_ss addsimps simp_thms 
wenzelm@23466
   108
  @ map (symmetric o mk_meta_eq) 
huffman@23469
   109
    [@{thm "dvd_eq_mod_eq_0"}, @{thm "zdvd_iff_zmod_eq_0"}, @{thm "mod_add1_eq"}, 
huffman@23469
   110
     @{thm "mod_add_left_eq"}, @{thm "mod_add_right_eq"}, 
wenzelm@23466
   111
     @{thm "zmod_zadd1_eq"}, @{thm "zmod_zadd_left_eq"}, 
wenzelm@23466
   112
     @{thm "zmod_zadd_right_eq"}, @{thm "div_add1_eq"}, @{thm "zdiv_zadd1_eq"}]
wenzelm@23466
   113
  @ [@{thm "mod_self"}, @{thm "zmod_self"}, @{thm "DIVISION_BY_ZERO_MOD"}, 
wenzelm@23466
   114
     @{thm "DIVISION_BY_ZERO_DIV"}, @{thm "DIVISION_BY_ZERO"} RS conjunct1, 
wenzelm@23466
   115
     @{thm "DIVISION_BY_ZERO"} RS conjunct2, @{thm "zdiv_zero"}, @{thm "zmod_zero"}, 
wenzelm@23466
   116
     @{thm "div_0"}, @{thm "mod_0"}, @{thm "zdiv_1"}, @{thm "zmod_1"}, @{thm "div_1"}, 
wenzelm@23466
   117
     @{thm "mod_1"}, @{thm "Suc_plus1"}]
wenzelm@23466
   118
  @ add_ac
wenzelm@23466
   119
 addsimprocs [cancel_div_mod_proc]
wenzelm@23466
   120
 val splits_ss = comp_ss addsimps [@{thm "mod_div_equality'"}] addsplits 
wenzelm@23466
   121
     [@{thm "split_zdiv"}, @{thm "split_zmod"}, @{thm "split_div'"}, 
wenzelm@23466
   122
      @{thm "split_min"}, @{thm "split_max"}, @{thm "abs_split"}]
wenzelm@23466
   123
in
wenzelm@23499
   124
fun nat_to_int_tac ctxt = 
wenzelm@23499
   125
  simp_tac (Simplifier.context ctxt ss1) THEN_ALL_NEW
wenzelm@23499
   126
  simp_tac (Simplifier.context ctxt ss2) THEN_ALL_NEW
wenzelm@23499
   127
  simp_tac (Simplifier.context ctxt comp_ss);
wenzelm@23466
   128
wenzelm@23499
   129
fun div_mod_tac ctxt i = simp_tac (Simplifier.context ctxt div_mod_ss) i;
wenzelm@23466
   130
fun splits_tac ctxt i = simp_tac (Simplifier.context ctxt splits_ss) i;
wenzelm@23466
   131
end;
wenzelm@23466
   132
wenzelm@23466
   133
wenzelm@23499
   134
fun core_cooper_tac ctxt = CSUBGOAL (fn (p, i) =>
wenzelm@23466
   135
   let 
wenzelm@23466
   136
    val cpth = 
wenzelm@23466
   137
       if !quick_and_dirty 
wenzelm@23466
   138
       then linzqe_oracle (ProofContext.theory_of ctxt) 
wenzelm@23466
   139
             (Envir.beta_norm (Pattern.eta_long [] (term_of (Thm.dest_arg p))))
wenzelm@23466
   140
       else arg_conv (Cooper.cooper_conv ctxt) p
wenzelm@23466
   141
    val p' = Thm.rhs_of cpth
wenzelm@23466
   142
    val th = implies_intr p' (equal_elim (symmetric cpth) (assume p'))
wenzelm@23499
   143
   in rtac th i end
wenzelm@23499
   144
   handle Cooper.COOPER _ => no_tac);
wenzelm@23466
   145
wenzelm@23499
   146
fun finish_tac q = SUBGOAL (fn (_, i) =>
wenzelm@23499
   147
  (if q then I else TRY) (rtac TrueI i));
wenzelm@23466
   148
wenzelm@23499
   149
fun cooper_tac elim add_ths del_ths ctxt =
wenzelm@23466
   150
let val ss = fst (CooperData.get ctxt) delsimps del_ths addsimps add_ths
wenzelm@23466
   151
in
wenzelm@23499
   152
  ObjectLogic.full_atomize_tac
wenzelm@23530
   153
  THEN_ALL_NEW CONVERSION Thm.eta_long_conversion
wenzelm@23499
   154
  THEN_ALL_NEW simp_tac ss
wenzelm@23499
   155
  THEN_ALL_NEW (TRY o generalize_tac (int_nat_terms ctxt))
wenzelm@23499
   156
  THEN_ALL_NEW ObjectLogic.full_atomize_tac
wenzelm@23499
   157
  THEN_ALL_NEW div_mod_tac ctxt
wenzelm@23499
   158
  THEN_ALL_NEW splits_tac ctxt
wenzelm@23499
   159
  THEN_ALL_NEW simp_tac ss
wenzelm@23530
   160
  THEN_ALL_NEW CONVERSION Thm.eta_long_conversion
wenzelm@23499
   161
  THEN_ALL_NEW nat_to_int_tac ctxt
wenzelm@23499
   162
  THEN_ALL_NEW (TRY o thin_prems_tac (is_relevant ctxt))
wenzelm@23499
   163
  THEN_ALL_NEW core_cooper_tac ctxt
wenzelm@23499
   164
  THEN_ALL_NEW finish_tac elim
wenzelm@23466
   165
end;
wenzelm@23466
   166
wenzelm@23466
   167
end;