discriminate.ml 11.5 KB
Newer Older
Andrei Paskevich's avatar
Andrei Paskevich committed
1 2 3 4 5 6 7 8 9 10
(********************************************************************)
(*                                                                  *)
(*  The Why3 Verification Platform   /   The Why3 Development Team  *)
(*  Copyright 2010-2012   --   INRIA - CNRS - Paris-Sud University  *)
(*                                                                  *)
(*  This software is distributed under the terms of the GNU Lesser  *)
(*  General Public License version 2.1, with the special exception  *)
(*  on linking described in file LICENSE.                           *)
(*                                                                  *)
(********************************************************************)
11 12 13 14 15 16 17 18 19

open Util
open Ident
open Ty
open Term
open Decl
open Theory
open Task

20
let meta_inst   = register_meta "encoding : inst"   [MTty]
21 22
  ~desc:"Specify@ which@ type@ should@ instantiate@ symbol@ marked@ by@ \
         'encoding : lskept'."
23
let meta_lskept = register_meta "encoding : lskept" [MTlsymbol]
24 25 26
  ~desc:"Specify@ which@ function@ symbols@ should@ be@ kept.@ If@ the@ \
         function@ is@ polymorphic,@ the@ monorphisations@ which@ signatures@ \
         contain@ only@ types@ @ marked@ with@ 'encoding : inst'@ are@ kept."
27
let meta_lsinst = register_meta "encoding : lsinst" [MTlsymbol;MTlsymbol]
28 29 30
  ~desc:"Specify@ which@ instantiations@ of@ symbols@ should@ be@ kept.@ \
         The first@ symbol@ specifies@ the@ symbol,@ the@ signature@ of@ the@ \
         second@ specifies@ the@ instantiation@ to@ keep."
31

32
let meta_select_inst   = register_meta_excl "select_inst"   [MTstring]
33 34 35 36 37 38 39 40 41
  ~desc:"@[Specify@ how@ to@ automatically@ mark@ type@ by@ \
'encoding : inst':@]@\n  \
@[\
  - none:@[ don't@ mark@ automatically@]@\n\
  - goal:@[ mark@ all@ the@ closed@ type@ that@ appear@ has@ argument@ \
            in@ the@ goal@]@\n\
  - all:@[ same@ as@ goal@ but@ also@ in@ the@ premises.@]\
@]"

42
let meta_select_lskept = register_meta_excl "select_lskept" [MTstring]
43 44 45 46 47 48 49 50 51 52
  ~desc:"@[Specify@ how@ to@ automatically@ mark@ symbol@ by@ \
'encoding : lskept':@]@\n  \
@[\
  - none:@[ don't@ mark@ automatically@]@\n\
  - goal:@[ mark@ all@ the@ symbols@ acceptable@ that@ appear@ in@ the@ \
            goal.@ The signature@ of an@ acceptable@ symbol@ contain@ at@ \
            least@ one@ type@ that@ is@ not@ a@ type@ variable@ neither@]@\n\
  - all:@[ same@ as@ goal@ but@ also@ in@ the@ premises.@]\
@]"

53
let meta_select_lsinst = register_meta_excl "select_lsinst" [MTstring]
54 55 56 57 58 59 60 61
  ~desc:"@[Specify@ how@ to@ automatically@ mark@ symbol@ by@ \
'encoding : lskept':@]@\n  \
@[\
  - none:@[ don't@ mark@ automatically@]@\n\
  - goal:@[ mark@ all@ the@ instantiation of symbols@ that@ appear@ in@ \
             the@ goal@]@\n\
  - all:@[ same@ as@ goal@ but@ also@ in@ the@ premises.@]\
@]"
62

63 64 65 66
module OHTy = OrderedHash(struct
  type t = ty
  let tag = ty_hash
end)
67

68 69 70 71
module OHTyl = OrderedHashList(struct
  type t = ty
  let tag = ty_hash
end)
72

73 74 75
module Mtyl = Stdlib.Map.Make(OHTyl)

module Lsmap = struct
76 77 78 79 80

(* TODO : transmettre les tags des logiques polymorphe vers les logiques
   instantié. Un tag sur un logique polymorphe doit être un tag sur toute
   la famille de fonctions *)

81
  type t = lsymbol Mtyl.t Mls.t
82

83
  let empty = Mls.empty
84

85 86
  let add ls tyl tyv lsmap =
    if ls_equal ls ps_equ then lsmap else
87
    if not (List.for_all Ty.ty_closed (oty_cons tyl tyv)) then lsmap else
88 89 90 91
    let newls = function
      | None -> Some (create_lsymbol (id_clone ls.ls_name) tyl tyv)
      | nls  -> nls
    in
92 93
    let insts = Mls.find_def Mtyl.empty ls lsmap in
    Mls.add ls (Mtyl.change newls (oty_cons tyl tyv) insts) lsmap
94

95
(* dead code
96 97 98 99 100 101
  let print_env fmt menv =
    Format.fprintf fmt "defined_lsymbol (%a)@."
      (Pp.print_iter2 Mls.iter Pp.semi Pp.comma Pretty.print_ls
         (Pp.print_iter2 Mtyl.iter Pp.semi Pp.arrow
            (Pp.print_list Pp.space Pretty.print_ty)
            Pretty.print_ls)) menv
102
*)
103 104

  (** From/To metas *)
105 106 107 108 109
  let metas lsmap =
    let fold_inst ls _ lsinst decls =
      create_meta meta_lsinst [MAls ls; MAls lsinst] :: decls in
    let fold_ls ls insts decls = Mtyl.fold (fold_inst ls) insts decls in
    Mls.fold fold_ls lsmap []
110

111
  let of_metas metas =
112 113
    let fold env args =
      match args with
114 115 116
        | [MAls ls; MAls lsinst] ->
          let tydisl = oty_cons lsinst.ls_args lsinst.ls_value in
          if not (List.for_all Ty.ty_closed tydisl) then env else
117
          let insts = Mls.find_def Mtyl.empty ls env in
118 119 120 121 122 123 124
          Mls.add ls (Mtyl.add tydisl lsinst insts) env
        | _ -> assert false
    in
    List.fold_left fold Mls.empty metas

end

125 126 127 128
let find_logic env ls tyl tyv =
  if ls_equal ls ps_equ then ls else
  try Mtyl.find (oty_cons tyl tyv) (Mls.find ls env)
  with Not_found -> ls
129

130 131 132
module Ssubst = Set.Make(struct
  type t = ty Mtv.t
  let compare = Mtv.compare OHTy.compare end)
133 134

(* find all the possible instantiation which can create a kept instantiation *)
135
let ty_quant env t =
136
  let add_vs acc0 ls tyl tyv =
137 138 139
    if ls_equal ls ps_equ then acc0 else
      try
        let insts = Mls.find ls env in
140
        let tyl = oty_cons tyl tyv in
141 142 143
        let fold_inst inst _ acc =
          let fold_subst subst acc =
            try
144
              let subst = List.fold_left2 ty_match subst tyl inst in
145 146 147 148 149 150 151 152 153
              Ssubst.add subst acc
            with TypeMismatch _ -> acc
          in
          (* fold on acc0 *)
          Ssubst.fold fold_subst acc0 acc
        in
        Mtyl.fold fold_inst insts acc0
      with Not_found (* no such p *) -> acc0
  in
154 155 156 157 158 159 160
  t_app_fold add_vs (Ssubst.singleton (Mtv.empty)) t

let ts_of_ls env ls decls =
  if ls_equal ls ps_equ then decls else
  let add_ts sts ts = Sts.add ts sts in
  let add_ty sts ty = ty_s_fold add_ts sts ty in
  let add_tyl tyl _ sts = List.fold_left add_ty sts tyl in
161
  let insts = Mls.find_def Mtyl.empty ls env in
162
  let sts = Mtyl.fold add_tyl insts Sts.empty in
163
  let add_ts ts dl = create_ty_decl ts :: dl in
164
  Sts.fold add_ts sts decls
165 166

(* The Core of the transformation *)
167
let map env d = match d.d_node with
168 169
  | Dtype _ -> [d]
  | Ddata _ -> Printer.unsupportedDecl d
170 171
      "Algebraic and recursively-defined types are \
            not supported, run eliminate_algebraic"
172
  | Dparam ls ->
173
      let lls = Mtyl.values (Mls.find_def Mtyl.empty ls env) in
174
      let lds = List.map create_param_decl lls in
175
      ts_of_ls env ls (d::lds)
176
  | Dlogic [ls,ld] when not (Sid.mem ls.ls_name d.d_syms) ->
177 178 179 180 181
      let f = ls_defn_axiom ld in
      let substs = ty_quant env f in
      let conv_f tvar (defns,axioms) =
        let f = t_ty_subst tvar Mvs.empty f in
        let f = t_app_map (find_logic env) f in
182 183 184 185 186 187 188
        match ls_defn_of_axiom f with
          | Some ld ->
              create_logic_decl [ld] :: defns, axioms
          | None ->
              let nm = ls.ls_name.id_string ^ "_inst" in
              let pr = create_prsymbol (id_derive nm ls.ls_name) in
              defns, create_prop_decl Paxiom pr f :: axioms
189 190 191 192
      in
      let defns,axioms = Ssubst.fold conv_f substs ([],[]) in
      ts_of_ls env ls (List.rev_append defns axioms)
  | Dlogic _ -> Printer.unsupportedDecl d
193
      "Recursively-defined symbols are not supported, run eliminate_recursion"
194
  | Dind _ -> Printer.unsupportedDecl d
195
      "Inductive predicates are not supported, run eliminate_inductive"
196
  | Dprop (k,pr,f) ->
197 198 199 200
      let substs = ty_quant env f in
      let substs_len = Ssubst.cardinal substs in
      let conv_f tvar task =
        (* Format.eprintf "f0 : %a@. env : %a@." Pretty.print_fmla *)
201
        (*   (t_ty_subst tvar Mvs.empty f) *)
202
        (*   print_env env; *)
203 204
        let f = t_ty_subst tvar Mvs.empty f in
        let f = t_app_map (find_logic env) f in
205 206 207 208 209 210 211 212 213 214 215 216 217 218 219
        (* Format.eprintf "f : %a@. env : %a@." Pretty.print_fmla f *)
        (*   print_env menv; *)
        let pr =
          if substs_len = 1 then pr
          else create_prsymbol (id_clone pr.pr_name) in
        (* Format.eprintf "undef ls : %a, ts : %a@." *)
        (*   (Pp.print_iter1 Sls.iter Pp.comma Pretty.print_ls) *)
        (*   menv.undef_lsymbol *)
        (*   (Pp.print_iter1 Sts.iter Pp.comma Pretty.print_ts) *)
        (*   menv.undef_tsymbol; *)
        create_prop_decl k pr f :: task
      in
      let task = Ssubst.fold conv_f substs [] in
      task

220 221

let ft_select_inst =
222
  ((Hstr.create 17) : (Env.env,Sty.t) Trans.flag_trans)
223 224

let ft_select_lskept =
225
  ((Hstr.create 17) : (Env.env,Sls.t) Trans.flag_trans)
226 227

let ft_select_lsinst =
228
  ((Hstr.create 17) : (Env.env,Lsmap.t) Trans.flag_trans)
229 230 231 232
let metas_from_env env =
  let fold_inst tyl _ s = List.fold_left (fun s ty -> Sty.add ty s) s tyl in
  let fold_ls _ insts s = Mtyl.fold fold_inst insts s in
  let sty = Mls.fold fold_ls env Sty.empty in
233
  let add ty decls = create_meta Libencoding.meta_kept [MAty ty] :: decls in
234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253
  Sty.fold add sty (Lsmap.metas env)

let lsinst_completion kept lskept env =
  let fold_ls ls env =
    let rec aux env tydisl subst = function
      | [] ->
          let tydisl = List.rev tydisl in
          let tyl,tyv = match tydisl, ls.ls_value with
            | tyv::tyl, Some _ -> tyl, Some tyv
            | tyl, None -> tyl, None
            | _ -> assert false in
          Lsmap.add ls tyl tyv env
      | ty::tyl ->
          let fold_ty tykept env =
            try
              let subst = ty_match subst ty tykept in
              aux env (tykept::tydisl) subst tyl
            with TypeMismatch _ -> env
          in
          Sty.fold fold_ty kept env
254
    in
255 256 257 258
    aux env [] Mtv.empty (oty_cons ls.ls_args ls.ls_value)
  in
  Sls.fold fold_ls lskept env

259 260 261 262 263 264 265 266 267
let add_user_lsinst env = function
  | [MAls ls; MAls nls] -> Lsmap.add ls nls.ls_args nls.ls_value env
  | _ -> assert false

let clear_metas = Trans.fold (fun hd task ->
  match hd.task_decl.td_node with
    | Meta (m,_) when meta_equal m meta_lsinst -> task
    | _ -> add_tdecl task hd.task_decl) None

268 269 270 271 272 273 274 275 276 277
let select_lsinst env =
  let inst   = Trans.on_flag meta_select_inst   ft_select_inst   "none" env in
  let lskept = Trans.on_flag meta_select_lskept ft_select_lskept "none" env in
  let lsinst = Trans.on_flag meta_select_lsinst ft_select_lsinst "none" env in
  let trans task =
    let inst   = Trans.apply inst   task in
    let lskept = Trans.apply lskept task in
    let lsinst = Trans.apply lsinst task in
    let inst   = Sty.union inst   (Task.on_tagged_ty meta_inst   task) in
    let lskept = Sls.union lskept (Task.on_tagged_ls meta_lskept task) in
278 279 280 281
    let lsinst = Task.on_meta meta_lsinst add_user_lsinst lsinst task in
    let lsinst = lsinst_completion inst lskept lsinst in
    let task   = Trans.apply clear_metas task in
    Trans.apply (Trans.add_tdecls (metas_from_env lsinst)) task
282
  in
283
  Trans.store trans
284 285 286 287 288

let lsymbol_distinction =
  Trans.on_meta meta_lsinst (fun metas ->
    if metas = [] then Trans.identity
    else
289
      let env = Lsmap.of_metas metas in
290
      (* Format.eprintf "instantiate %a@." print_env env; *)
291 292 293
      Trans.decl (map env) None)

let discriminate env = Trans.seq [
294
  Libencoding.monomorphise_goal;
295
  select_lsinst env;
296
  Trans.print_meta Libencoding.debug meta_lsinst;
297 298 299
  lsymbol_distinction;
]

300 301
let empty_formatted : Pp.formatted = ""

302
let () = Trans.register_env_transform "discriminate" discriminate
303 304 305 306 307 308 309 310 311
  ~desc_metas:[meta_select_inst  , empty_formatted;
               meta_select_lskept, empty_formatted;
               meta_select_lsinst, empty_formatted;
               meta_inst         , empty_formatted;
               meta_lskept       , empty_formatted;
               meta_lsinst       , empty_formatted]
  ~desc:"Discriminate@ polymorphic@ function@ and@ predicate@ symbols.@ \
         Allow@ to@ keep@ some@ instantiations@ from@ being@ touched@ by@ \
         following@ polymorphism@ encodings."
312