(************************************************************************) (* v * The Coq Proof Assistant / The Coq Development Team *) (* None | Some resolve -> try List.assoc kn resolve with Not_found -> None type substitution_domain = MSI of mod_self_id | MBI of mod_bound_id | MPI of module_path let string_of_subst_domain = function MSI msid -> debug_string_of_msid msid | MBI mbid -> debug_string_of_mbid mbid | MPI mp -> string_of_mp mp module Umap = Map.Make(struct type t = substitution_domain let compare = Pervasives.compare end) type substitution = (module_path * resolver option) Umap.t let empty_subst = Umap.empty let add_msid msid mp = Umap.add (MSI msid) (mp,None) let add_mbid mbid mp resolve = Umap.add (MBI mbid) (mp,resolve) let add_mp mp1 mp2 = Umap.add (MPI mp1) (mp2,None) let map_msid msid mp = add_msid msid mp empty_subst let map_mbid mbid mp resolve = add_mbid mbid mp resolve empty_subst let map_mp mp1 mp2 = add_mp mp1 mp2 empty_subst let list_contents sub = let one_pair uid (mp,_) l = (string_of_subst_domain uid, string_of_mp mp)::l in Umap.fold one_pair sub [] let debug_string_of_subst sub = let l = List.map (fun (s1,s2) -> s1^"|->"^s2) (list_contents sub) in "{" ^ String.concat "; " l ^ "}" let debug_pr_subst sub = let l = list_contents sub in let f (s1,s2) = hov 2 (str s1 ++ spc () ++ str "|-> " ++ str s2) in str "{" ++ hov 2 (prlist_with_sep pr_coma f l) ++ str "}" let subst_mp0 sub mp = (* 's like subst *) let rec aux mp = match mp with | MPself sid -> let mp',resolve = Umap.find (MSI sid) sub in mp',resolve | MPbound bid -> let mp',resolve = Umap.find (MBI bid) sub in mp',resolve | MPdot (mp1,l) as mp2 -> begin try let mp',resolve = Umap.find (MPI mp2) sub in mp',resolve with Not_found -> let mp1',resolve = aux mp1 in MPdot (mp1',l),resolve end | _ -> raise Not_found in try Some (aux mp) with Not_found -> None let subst_mp sub mp = match subst_mp0 sub mp with None -> mp | Some (mp',_) -> mp' let subst_kn0 sub kn = let mp,dir,l = repr_kn kn in match subst_mp0 sub mp with Some (mp',_) -> Some (make_kn mp' dir l) | None -> None let subst_kn sub kn = match subst_kn0 sub kn with None -> kn | Some kn' -> kn' let subst_con sub con = let mp,dir,l = repr_con con in match subst_mp0 sub mp with None -> con,mkConst con | Some (mp',resolve) -> let con' = make_con mp' dir l in match apply_opt_resolver resolve con with None -> con',mkConst con' | Some t -> con',t let subst_con0 sub con = let mp,dir,l = repr_con con in match subst_mp0 sub mp with None -> None | Some (mp',resolve) -> let con' = make_con mp' dir l in match apply_opt_resolver resolve con with None -> Some (mkConst con') | Some t -> Some t (* Here the semantics is completely unclear. What does "Hint Unfold t" means when "t" is a parameter? Does the user mean "Unfold X.t" or does she mean "Unfold y" where X.t is later on instantiated with y? I choose the first interpretation (i.e. an evaluable reference is never expanded). *) let subst_evaluable_reference subst = function | EvalVarRef id -> EvalVarRef id | EvalConstRef kn -> EvalConstRef (fst (subst_con subst kn)) let rec map_kn f f' c = let func = map_kn f f' in match kind_of_term c with | Const kn -> (match f' kn with None -> c | Some const ->const) | Ind (kn,i) -> (match f kn with None -> c | Some kn' -> mkInd (kn',i)) | Construct ((kn,i),j) -> (match f kn with None -> c | Some kn' -> mkConstruct ((kn',i),j)) | Case (ci,p,ct,l) -> let ci_ind = let (kn,i) = ci.ci_ind in (match f kn with None -> ci.ci_ind | Some kn' -> kn',i ) in let p' = func p in let ct' = func ct in let l' = array_smartmap func l in if (ci.ci_ind==ci_ind && p'==p && l'==l && ct'==ct)then c else mkCase ({ci with ci_ind = ci_ind}, p',ct', l') | Cast (ct,k,t) -> let ct' = func ct in let t'= func t in if (t'==t && ct'==ct) then c else mkCast (ct', k, t') | Prod (na,t,ct) -> let ct' = func ct in let t'= func t in if (t'==t && ct'==ct) then c else mkProd (na, t', ct') | Lambda (na,t,ct) -> let ct' = func ct in let t'= func t in if (t'==t && ct'==ct) then c else mkLambda (na, t', ct') | LetIn (na,b,t,ct) -> let ct' = func ct in let t'= func t in let b'= func b in if (t'==t && ct'==ct && b==b') then c else mkLetIn (na, b', t', ct') | App (ct,l) -> let ct' = func ct in let l' = array_smartmap func l in if (ct'== ct && l'==l) then c else mkApp (ct',l') | Evar (e,l) -> let l' = array_smartmap func l in if (l'==l) then c else mkEvar (e,l') | Fix (ln,(lna,tl,bl)) -> let tl' = array_smartmap func tl in let bl' = array_smartmap func bl in if (bl == bl'&& tl == tl') then c else mkFix (ln,(lna,tl',bl')) | CoFix(ln,(lna,tl,bl)) -> let tl' = array_smartmap func tl in let bl' = array_smartmap func bl in if (bl == bl'&& tl == tl') then c else mkCoFix (ln,(lna,tl',bl')) | _ -> c let subst_mps sub = map_kn (subst_kn0 sub) (subst_con0 sub) let rec replace_mp_in_mp mpfrom mpto mp = match mp with | _ when mp = mpfrom -> mpto | MPdot (mp1,l) -> let mp1' = replace_mp_in_mp mpfrom mpto mp1 in if mp1==mp1' then mp else MPdot (mp1',l) | _ -> mp let replace_mp_in_con mpfrom mpto kn = let mp,dir,l = repr_con kn in let mp'' = replace_mp_in_mp mpfrom mpto mp in if mp==mp'' then kn else make_con mp'' dir l exception BothSubstitutionsAreIdentitySubstitutions exception ChangeDomain of resolver let join (subst1 : substitution) (subst2 : substitution) = let apply_subst (sub : substitution) key (mp,resolve) = let mp',resolve' = match subst_mp0 sub mp with None -> mp, None | Some (mp',resolve') -> mp',resolve' in let resolve'' : resolver option = try let res = match resolve with Some res -> res | None -> match resolve' with None -> raise BothSubstitutionsAreIdentitySubstitutions | Some res -> raise (ChangeDomain res) in Some (List.map (fun (kn,topt) -> kn, match topt with None -> (match key with MSI msid -> let kn' = replace_mp_in_con (MPself msid) mp kn in apply_opt_resolver resolve' kn' | MBI mbid -> let kn' = replace_mp_in_con (MPbound mbid) mp kn in apply_opt_resolver resolve' kn' | MPI mp1 -> let kn' = replace_mp_in_con mp1 mp kn in apply_opt_resolver resolve' kn') | Some t -> Some (subst_mps sub t)) res) with BothSubstitutionsAreIdentitySubstitutions -> None | ChangeDomain res -> let rec changeDom = function | [] -> [] | (kn,topt)::r -> let key' = match key with MSI msid -> MPself msid | MBI mbid -> MPbound mbid | MPI mp1 -> mp1 in let kn' = replace_mp_in_con mp key' kn in if kn==kn' then (*the key does not appear in kn, we remove it from the resolver that we are building*) changeDom r else (kn',topt)::(changeDom r) in Some (changeDom res) in mp',resolve'' in let subst = Umap.mapi (apply_subst subst2) subst1 in (Umap.fold Umap.add subst2 subst) let subst_key subst1 subst2 = let replace_in_key key (mp,resolve) sub= let newkey = match key with | MPI mp1 -> begin match subst_mp0 subst1 mp1 with | None -> None | Some (mp2,_) -> Some (MPI mp2) end | _ -> None in match newkey with | None -> Umap.add key (mp,resolve) sub | Some mpi -> Umap.add mpi (mp,resolve) sub in Umap.fold replace_in_key subst2 empty_subst let update_subst_alias subst1 subst2 = let subst_inv key (mp,resolve) sub = let newmp = match key with | MBI msid -> MPbound msid | MSI msid -> MPself msid | MPI mp -> mp in match mp with | MPbound mbid -> Umap.add (MBI mbid) (newmp,None) sub | MPself msid -> Umap.add (MSI msid) (newmp,None) sub | _ -> Umap.add (MPI mp) (newmp,None) sub in let subst_mbi = Umap.fold subst_inv subst2 empty_subst in let alias_subst key (mp,resolve) sub= let newkey = match key with | MPI mp1 -> begin match subst_mp0 subst_mbi mp1 with | None -> None | Some (mp2,_) -> Some (MPI mp2) end | _ -> None in match newkey with | None -> Umap.add key (mp,resolve) sub | Some mpi -> Umap.add mpi (mp,resolve) sub in Umap.fold alias_subst subst1 empty_subst let update_subst subst1 subst2 = let subst_inv key (mp,resolve) l = let newmp = match key with | MBI msid -> MPbound msid | MSI msid -> MPself msid | MPI mp -> mp in match mp with | MPbound mbid -> ((MBI mbid),newmp)::l | MPself msid -> ((MSI msid),newmp)::l | _ -> ((MPI mp),newmp)::l in let subst_mbi = Umap.fold subst_inv subst2 [] in let alias_subst key (mp,resolve) sub= let newsetkey = match key with | MPI mp1 -> let compute_set_newkey l (k,mp') = let mp_from_key = match k with | MBI msid -> MPbound msid | MSI msid -> MPself msid | MPI mp -> mp in let new_mp1 = replace_mp_in_mp mp_from_key mp' mp1 in if new_mp1 == mp1 then l else (MPI new_mp1)::l in begin match List.fold_left compute_set_newkey [] subst_mbi with | [] -> None | l -> Some (l) end | _ -> None in match newsetkey with | None -> sub | Some l -> List.fold_left (fun s k -> Umap.add k (mp,resolve) s) sub l in Umap.fold alias_subst subst1 empty_subst let join_alias (subst1 : substitution) (subst2 : substitution) = let apply_subst (sub : substitution) key (mp,resolve) = let mp',resolve' = match subst_mp0 sub mp with None -> mp, None | Some (mp',resolve') -> mp',resolve' in let resolve'' : resolver option = try let res = match resolve with Some res -> res | None -> match resolve' with None -> raise BothSubstitutionsAreIdentitySubstitutions | Some res -> raise (ChangeDomain res) in Some (List.map (fun (kn,topt) -> kn, match topt with None -> (match key with MSI msid -> let kn' = replace_mp_in_con (MPself msid) mp kn in apply_opt_resolver resolve' kn' | MBI mbid -> let kn' = replace_mp_in_con (MPbound mbid) mp kn in apply_opt_resolver resolve' kn' | MPI mp1 -> let kn' = replace_mp_in_con mp1 mp kn in apply_opt_resolver resolve' kn') | Some t -> Some (subst_mps sub t)) res) with BothSubstitutionsAreIdentitySubstitutions -> None | ChangeDomain res -> let rec changeDom = function | [] -> [] | (kn,topt)::r -> let key' = match key with MSI msid -> MPself msid | MBI mbid -> MPbound mbid | MPI mp1 -> mp1 in let kn' = replace_mp_in_con mp key' kn in if kn==kn' then (*the key does not appear in kn, we remove it from the resolver that we are building*) changeDom r else (kn',topt)::(changeDom r) in Some (changeDom res) in mp',resolve'' in Umap.mapi (apply_subst subst2) subst1 let remove_alias subst = let rec remove key (mp,resolve) sub = match key with MPI _ -> sub | _ -> Umap.add key (mp,resolve) sub in Umap.fold remove subst empty_subst let rec occur_in_path uid path = match uid,path with | MSI sid,MPself sid' -> sid = sid' | MBI bid,MPbound bid' -> bid = bid' | _,MPdot (mp1,_) -> occur_in_path uid mp1 | _ -> false let occur_uid uid sub = let check_one uid' (mp,_) = if uid = uid' || occur_in_path uid mp then raise Exit in try Umap.iter check_one sub; false with Exit -> true let occur_msid uid = occur_uid (MSI uid) let occur_mbid uid = occur_uid (MBI uid) type 'a lazy_subst = | LSval of 'a | LSlazy of substitution * 'a type 'a substituted = 'a lazy_subst ref let from_val a = ref (LSval a) let force fsubst r = match !r with | LSval a -> a | LSlazy(s,a) -> let a' = fsubst s a in r := LSval a'; a' let subst_substituted s r = match !r with | LSval a -> ref (LSlazy(s,a)) | LSlazy(s',a) -> let s'' = join s' s in ref (LSlazy(s'',a))