discriminate.ml 10.1 KB
Newer Older
1 2
(**************************************************************************)
(*                                                                        *)
MARCHE Claude's avatar
MARCHE Claude committed
3
(*  Copyright (C) 2010-2012                                               *)
4 5 6
(*    François Bobot                                                      *)
(*    Jean-Christophe Filliâtre                                           *)
(*    Claude Marché                                                       *)
MARCHE Claude's avatar
MARCHE Claude committed
7
(*    Guillaume Melquiond                                                 *)
8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28
(*    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.                  *)
(*                                                                        *)
(**************************************************************************)

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

29 30 31
let meta_inst   = register_meta "encoding : inst"   [MTty]
let meta_lskept = register_meta "encoding : lskept" [MTlsymbol]
let meta_lsinst = register_meta "encoding : lsinst" [MTlsymbol;MTlsymbol]
32

33 34 35
let meta_select_inst   = register_meta_excl "select_inst"   [MTstring]
let meta_select_lskept = register_meta_excl "select_lskept" [MTstring]
let meta_select_lsinst = register_meta_excl "select_lsinst" [MTstring]
36

37 38 39 40
module OHTy = OrderedHash(struct
  type t = ty
  let tag = ty_hash
end)
41

42 43 44 45
module OHTyl = OrderedHashList(struct
  type t = ty
  let tag = ty_hash
end)
46

47 48 49
module Mtyl = Stdlib.Map.Make(OHTyl)

module Lsmap = struct
50 51 52 53 54

(* 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 *)

55
  type t = lsymbol Mtyl.t Mls.t
56

57
  let empty = Mls.empty
58

59 60
  let add ls tyl tyv lsmap =
    if ls_equal ls ps_equ then lsmap else
61
    if not (List.for_all Ty.ty_closed (oty_cons tyl tyv)) then lsmap else
62 63 64 65
    let newls = function
      | None -> Some (create_lsymbol (id_clone ls.ls_name) tyl tyv)
      | nls  -> nls
    in
66 67
    let insts = Mls.find_def Mtyl.empty ls lsmap in
    Mls.add ls (Mtyl.change newls (oty_cons tyl tyv) insts) lsmap
68 69 70 71 72 73 74 75 76

  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

  (** From/To metas *)
77 78 79 80 81
  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 []
82

83
  let of_metas metas =
84 85
    let fold env args =
      match args with
86 87 88
        | [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
89
          let insts = Mls.find_def Mtyl.empty ls env in
90 91 92 93 94 95 96
          Mls.add ls (Mtyl.add tydisl lsinst insts) env
        | _ -> assert false
    in
    List.fold_left fold Mls.empty metas

end

97 98 99 100
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
101

102 103 104
module Ssubst = Set.Make(struct
  type t = ty Mtv.t
  let compare = Mtv.compare OHTy.compare end)
105 106

(* find all the possible instantiation which can create a kept instantiation *)
107
let ty_quant env t =
108 109 110 111
  let rec add_vs acc0 ls tyl tyv =
    if ls_equal ls ps_equ then acc0 else
      try
        let insts = Mls.find ls env in
112
        let tyl = oty_cons tyl tyv in
113 114 115
        let fold_inst inst _ acc =
          let fold_subst subst acc =
            try
116
              let subst = List.fold_left2 ty_match subst tyl inst in
117 118 119 120 121 122 123 124 125
              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
126 127 128 129 130 131 132
  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
133
  let insts = Mls.find_def Mtyl.empty ls env in
134
  let sts = Mtyl.fold add_tyl insts Sts.empty in
135
  let add_ts ts dl = create_ty_decl ts :: dl in
136
  Sts.fold add_ts sts decls
137 138

(* The Core of the transformation *)
139
let map env d = match d.d_node with
140 141
  | Dtype _ -> [d]
  | Ddata _ -> Printer.unsupportedDecl d
142 143
      "Algebraic and recursively-defined types are \
            not supported, run eliminate_algebraic"
144
  | Dparam ls ->
145
      let lls = Mtyl.values (Mls.find_def Mtyl.empty ls env) in
146
      let lds = List.map create_param_decl lls in
147
      ts_of_ls env ls (d::lds)
148
  | Dlogic [ls,ld] when not (Sid.mem ls.ls_name d.d_syms) ->
149 150 151 152 153
      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
154 155 156 157 158 159 160
        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
161 162 163 164
      in
      let defns,axioms = Ssubst.fold conv_f substs ([],[]) in
      ts_of_ls env ls (List.rev_append defns axioms)
  | Dlogic _ -> Printer.unsupportedDecl d
165
      "Recursively-defined symbols are not supported, run eliminate_recursion"
166
  | Dind _ -> Printer.unsupportedDecl d
167
      "Inductive predicates are not supported, run eliminate_inductive"
168
  | Dprop (k,pr,f) ->
169 170 171 172
      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 *)
173
        (*   (t_ty_subst tvar Mvs.empty f) *)
174
        (*   print_env env; *)
175 176
        let f = t_ty_subst tvar Mvs.empty f in
        let f = t_app_map (find_logic env) f in
177 178 179 180 181 182 183 184 185 186 187 188 189 190 191
        (* 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

192 193 194 195 196 197 198 199 200 201 202 203 204 205

let ft_select_inst =
  ((Hashtbl.create 17) : (Env.env,Sty.t) Trans.flag_trans)

let ft_select_lskept =
  ((Hashtbl.create 17) : (Env.env,Sls.t) Trans.flag_trans)

let ft_select_lsinst =
  ((Hashtbl.create 17) : (Env.env,Lsmap.t) Trans.flag_trans)

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
206
  let add ty decls = create_meta Libencoding.meta_kept [MAty ty] :: decls in
207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226
  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
227
    in
228 229 230 231
    aux env [] Mtv.empty (oty_cons ls.ls_args ls.ls_value)
  in
  Sls.fold fold_ls lskept env

232 233 234 235 236 237 238 239 240
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

241 242 243 244 245 246 247 248 249 250
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
251 252 253 254
    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
255
  in
256
  Trans.store trans
257 258 259 260 261

let lsymbol_distinction =
  Trans.on_meta meta_lsinst (fun metas ->
    if metas = [] then Trans.identity
    else
262
      let env = Lsmap.of_metas metas in
263
      (* Format.eprintf "instantiate %a@." print_env env; *)
264 265 266
      Trans.decl (map env) None)

let discriminate env = Trans.seq [
267
  Libencoding.monomorphise_goal;
268
  select_lsinst env;
269
  Trans.print_meta Libencoding.debug meta_lsinst;
270 271 272 273 274
  lsymbol_distinction;
]

let () = Trans.register_env_transform "discriminate" discriminate