lift_epsilon.ml 3.42 KB
Newer Older
1 2
(**************************************************************************)
(*                                                                        *)
MARCHE Claude's avatar
MARCHE Claude committed
3
(*  Copyright (C) 2010-2012                                               *)
4 5
(*    François Bobot                                                      *)
(*    Jean-Christophe Filliâtre                                           *)
6
(*    Johannes Kanig                                                      *)
7
(*    Claude Marché                                                       *)
8 9 10 11 12 13 14 15 16 17 18 19 20
(*    Andrei Paskevich                                                    *)
(*                                                                        *)
(*  This software is free software; you can redistribute it and/or        *)
(*  modify it under the terms of the GNU Library General Public           *)
(*  License version 2.1, with the special exception on linking            *)
(*  described in file LICENSE.                                            *)
(*                                                                        *)
(*  This software is distributed in the hope that it will be useful,      *)
(*  but WITHOUT ANY WARRANTY; without even the implied warranty of        *)
(*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.                  *)
(*                                                                        *)
(**************************************************************************)

21
open Close_epsilon
22 23 24 25
open Term
open Theory
open Task

26 27 28 29
type lift_kind =
(*   | Goal (* prove the existence of a witness *) *)
  | Implied (* require the existence of a witness in an axiom *)
  | Implicit (* do not require a witness *)
30

31 32 33 34
let lift kind =
  let rec term acc t =
    match t.t_node with
    | Teps fb ->
35
        let fv = Mvs.keys t.t_vars in
36
        let x, f = t_open_bound fb in
37 38 39 40
        let acc, f = form acc f in
        let tys = List.map (fun x -> x.vs_ty) fv in
        let xs = Ident.id_derive "epsilon" x.vs_name in
        let xl = create_fsymbol xs tys x.vs_ty in
41
        let acc = add_decl acc (Decl.create_param_decl xl) in
42 43
        let axs =
          Decl.create_prsymbol (Ident.id_derive ("epsilon_def") x.vs_name) in
44
        let xlapp = t_app xl (List.map t_var fv) t.t_ty in
45 46 47 48
        let f =
          match kind with
          (* assume that lambdas always exist *)
          | Implied when not (is_lambda t) ->
49 50
              t_forall_close_merge fv
                (t_implies (t_exists_close [x] [] f)
51 52
                   (t_subst_single x xlapp f))
          | _ -> t_subst_single x xlapp f
53 54 55
        in
        let acc = add_decl acc (Decl.create_prop_decl Decl.Paxiom axs f) in
        acc, xlapp
56 57
    | _ -> TermTF.t_map_fold term form acc t
  and form acc f = TermTF.t_map_fold term form acc f in
58 59 60 61
  fun th acc ->
    let th = th.task_decl in
    match th.td_node with
    | Decl d ->
62
        let acc, d = Decl.DeclTF.decl_map_fold term form acc d in
63 64
        add_decl acc d
    | _ -> add_tdecl acc th
65

Francois Bobot's avatar
Francois Bobot committed
66
let lift_epsilon kind = Trans.fold (lift kind) None
67

Francois Bobot's avatar
Francois Bobot committed
68
let meta_epsilon = Theory.register_meta_excl "lift_epsilon" [MTstring]
69
  ~desc:"TODO"
Francois Bobot's avatar
Francois Bobot committed
70

71 72 73 74 75
let lift_epsilon = Trans.on_meta_excl meta_epsilon
  (fun alo ->
    let kind = match alo with
      | Some [MAstr "implicit"] -> Implicit
      | Some [MAstr "implied"] | None -> Implied
Francois Bobot's avatar
Francois Bobot committed
76 77 78
      | _ -> failwith "lift_epsilon accepts only 'implicit' and 'implied'"
    in
    lift_epsilon kind)
79

Francois Bobot's avatar
Francois Bobot committed
80
let () = Trans.register_transform "lift_epsilon" lift_epsilon
81
  ~desc:"TODO"