src/HOL/Library/Eval_Witness.thy
author blanchet
Mon, 09 Aug 2010 12:53:16 +0200
changeset 38287 796302ca3611
parent 34028 1e6206763036
child 38549 d0385f2764d8
permissions -rw-r--r--
replace "setup" with "declaration"
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
24281
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
     1
(*  Title:      HOL/Library/Eval_Witness.thy
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
     2
    Author:     Alexander Krauss, TU Muenchen
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
     3
*)
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
     4
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
     5
header {* Evaluation Oracle with ML witnesses *}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
     6
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
     7
theory Eval_Witness
30663
0b6aff7451b2 Main is (Complex_Main) base entry point in library theories
haftmann
parents: 30549
diff changeset
     8
imports List Main
24281
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
     9
begin
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    10
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    11
text {* 
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    12
  We provide an oracle method similar to "eval", but with the
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    13
  possibility to provide ML values as witnesses for existential
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    14
  statements.
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    15
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    16
  Our oracle can prove statements of the form @{term "EX x. P x"}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    17
  where @{term "P"} is an executable predicate that can be compiled to
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    18
  ML. The oracle generates code for @{term "P"} and applies
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    19
  it to a user-specified ML value. If the evaluation
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    20
  returns true, this is effectively a proof of  @{term "EX x. P x"}.
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    21
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    22
  However, this is only sound if for every ML value of the given type
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    23
  there exists a corresponding HOL value, which could be used in an
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    24
  explicit proof. Unfortunately this is not true for function types,
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    25
  since ML functions are not equivalent to the pure HOL
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    26
  functions. Thus, the oracle can only be used on first-order
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    27
  types.
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    28
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    29
  We define a type class to mark types that can be safely used
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    30
  with the oracle.  
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    31
*}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    32
29608
564ea783ace8 no base sort in class import
haftmann
parents: 28290
diff changeset
    33
class ml_equiv
24281
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    34
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    35
text {*
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    36
  Instances of @{text "ml_equiv"} should only be declared for those types,
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    37
  where the universe of ML values coincides with the HOL values.
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    38
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    39
  Since this is essentially a statement about ML, there is no
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    40
  logical characterization.
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    41
*}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    42
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    43
instance nat :: ml_equiv .. (* Attention: This conflicts with the "EfficientNat" theory *)
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    44
instance bool :: ml_equiv ..
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    45
instance list :: (ml_equiv) ml_equiv ..
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    46
27103
d8549f4d900b major refactorings in code generator modules
haftmann
parents: 26939
diff changeset
    47
ML {*
d8549f4d900b major refactorings in code generator modules
haftmann
parents: 26939
diff changeset
    48
structure Eval_Witness_Method =
d8549f4d900b major refactorings in code generator modules
haftmann
parents: 26939
diff changeset
    49
struct
d8549f4d900b major refactorings in code generator modules
haftmann
parents: 26939
diff changeset
    50
32740
9dd0a2f83429 explicit indication of Unsynchronized.ref;
wenzelm
parents: 30970
diff changeset
    51
val eval_ref : (unit -> bool) option Unsynchronized.ref = Unsynchronized.ref NONE;
27103
d8549f4d900b major refactorings in code generator modules
haftmann
parents: 26939
diff changeset
    52
d8549f4d900b major refactorings in code generator modules
haftmann
parents: 26939
diff changeset
    53
end;
d8549f4d900b major refactorings in code generator modules
haftmann
parents: 26939
diff changeset
    54
*}
d8549f4d900b major refactorings in code generator modules
haftmann
parents: 26939
diff changeset
    55
28290
4cc2b6046258 simplified oracle interface;
wenzelm
parents: 28054
diff changeset
    56
oracle eval_witness_oracle = {* fn (cgoal, ws) =>
24281
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    57
let
28290
4cc2b6046258 simplified oracle interface;
wenzelm
parents: 28054
diff changeset
    58
  val thy = Thm.theory_of_cterm cgoal;
4cc2b6046258 simplified oracle interface;
wenzelm
parents: 28054
diff changeset
    59
  val goal = Thm.term_of cgoal;
24281
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    60
  fun check_type T = 
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    61
    if Sorts.of_sort (Sign.classes_of thy) (T, ["Eval_Witness.ml_equiv"])
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    62
    then T
26939
1035c89b4c02 moved global pretty/string_of functions from Sign to Syntax;
wenzelm
parents: 26114
diff changeset
    63
    else error ("Type " ^ quote (Syntax.string_of_typ_global thy T) ^ " not allowed for ML witnesses")
24281
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    64
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    65
  fun dest_exs  0 t = t
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    66
    | dest_exs n (Const ("Ex", _) $ Abs (v,T,b)) = 
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    67
      Abs (v, check_type T, dest_exs (n - 1) b)
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    68
    | dest_exs _ _ = sys_error "dest_exs";
24835
8c26128f8997 clarified relationship of code generator conversions and evaluations
haftmann
parents: 24281
diff changeset
    69
  val t = dest_exs (length ws) (HOLogic.dest_Trueprop goal);
24281
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    70
in
34028
1e6206763036 split off evaluation mechanisms in separte module Code_Eval
haftmann
parents: 32740
diff changeset
    71
  if Code_Eval.eval NONE ("Eval_Witness_Method.eval_ref", Eval_Witness_Method.eval_ref) (K I) thy t ws
28290
4cc2b6046258 simplified oracle interface;
wenzelm
parents: 28054
diff changeset
    72
  then Thm.cterm_of thy goal
4cc2b6046258 simplified oracle interface;
wenzelm
parents: 28054
diff changeset
    73
  else @{cprop True} (*dummy*)
24281
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    74
end
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    75
*}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    76
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    77
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    78
method_setup eval_witness = {*
30549
d2d7874648bd simplified method setup;
wenzelm
parents: 30510
diff changeset
    79
  Scan.lift (Scan.repeat Args.name) >>
d2d7874648bd simplified method setup;
wenzelm
parents: 30510
diff changeset
    80
  (fn ws => K (SIMPLE_METHOD'
d2d7874648bd simplified method setup;
wenzelm
parents: 30510
diff changeset
    81
    (CSUBGOAL (fn (ct, i) => rtac (eval_witness_oracle (ct, ws)) i))))
d2d7874648bd simplified method setup;
wenzelm
parents: 30510
diff changeset
    82
*} "evaluation with ML witnesses"
24281
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    83
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    84
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    85
subsection {* Toy Examples *}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    86
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    87
text {* 
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    88
  Note that we must use the generated data structure for the
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    89
  naturals, since ML integers are different.
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    90
*}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    91
26114
53eb3ff08cce non-operative code antiquotation
haftmann
parents: 25595
diff changeset
    92
(*lemma "\<exists>n::nat. n = 1"
53eb3ff08cce non-operative code antiquotation
haftmann
parents: 25595
diff changeset
    93
apply (eval_witness "Suc Zero_nat")
53eb3ff08cce non-operative code antiquotation
haftmann
parents: 25595
diff changeset
    94
done*)
24281
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    95
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    96
text {* 
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    97
  Since polymorphism is not allowed, we must specify the
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    98
  type explicitly:
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
    99
*}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   100
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   101
lemma "\<exists>l. length (l::bool list) = 3"
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   102
apply (eval_witness "[true,true,true]")
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   103
done
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   104
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   105
text {* Multiple witnesses *}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   106
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   107
lemma "\<exists>k l. length (k::bool list) = length (l::bool list)"
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   108
apply (eval_witness "[]" "[]")
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   109
done
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   110
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   111
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   112
subsection {* Discussion *}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   113
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   114
subsubsection {* Conflicts *}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   115
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   116
text {* 
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   117
  This theory conflicts with EfficientNat, since the @{text ml_equiv} instance
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   118
  for natural numbers is not valid when they are mapped to ML
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   119
  integers. With that theory loaded, we could use our oracle to prove
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   120
  @{term "\<exists>n. n < 0"} by providing @{text "~1"} as a witness.
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   121
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   122
  This shows that @{text ml_equiv} declarations have to be used with care,
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   123
  taking the configuration of the code generator into account.
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   124
*}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   125
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   126
subsubsection {* Haskell *}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   127
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   128
text {*
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   129
  If we were able to run generated Haskell code, the situation would
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   130
  be much nicer, since Haskell functions are pure and could be used as
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   131
  witnesses for HOL functions: Although Haskell functions are partial,
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   132
  we know that if the evaluation terminates, they are ``sufficiently
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   133
  defined'' and could be completed arbitrarily to a total (HOL) function.
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   134
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   135
  This would allow us to provide access to very efficient data
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   136
  structures via lookup functions coded in Haskell and provided to HOL
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   137
  as witnesses. 
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   138
*}
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   139
7d0334b69711 added Eval_Witness theory
haftmann
parents:
diff changeset
   140
end