(************************************************************************) (* * The Rocq Prover / The Rocq Development Team *) (* v * Copyright INRIA, CNRS and contributors *) (* <O___,, * (see version control and CREDITS file for authors & dates) *) (* \VV/ **************************************************************) (* // * This file is distributed under the terms of the *) (* * GNU Lesser General Public License Version 2.1 *) (* * (see LICENSE file for the text of the license) *) (************************************************************************)
open Pp open CErrors open Util open Names open Nameops open Term open Constr open Context open Vars open Environ
let debug_print_constr sigma c = Constr.debug_print (EConstr.to_constr sigma c) let fallback_printer _env sigma c = debug_print_constr sigma c let term_printer = ref fallback_printer
let print_constr_env env sigma t = !term_printer (env:env) sigma (t:Evd.econstr) let set_print_constr f = term_printer := f
let pr_var_decl env sigma decl = letopen NamedDecl in let pbody = match decl with
| LocalAssum _ -> mt ()
| LocalDef (_,c,_) -> (* Force evaluation *) let c = EConstr.of_constr c in let pb = print_constr_env env sigma c in
(str" := " ++ pb ++ cut () ) in let pt = print_constr_env env sigma (EConstr.of_constr (get_type decl)) in let ptyp = (str" : " ++ pt) in
(Id.print (get_id decl) ++ hov 0 (pbody ++ ptyp))
let pr_rel_decl env sigma decl = letopen RelDecl in let pbody = match decl with
| LocalAssum _ -> mt ()
| LocalDef (_,c,_) -> (* Force evaluation *) let c = EConstr.of_constr c in let pb = print_constr_env env sigma c in
(str":=" ++ spc () ++ pb ++ spc ()) in let ptyp = print_constr_env env sigma (EConstr.of_constr (get_type decl)) in match get_name decl with
| Anonymous -> hov 0 (str"<>" ++ spc () ++ pbody ++ str":" ++ spc () ++ ptyp)
| Name id -> hov 0 (Id.print id ++ spc () ++ pbody ++ str":" ++ spc () ++ ptyp)
let print_env env sigma = let sign_env =
fold_named_context
(fun env d pps -> let pidt = pr_var_decl env sigma d in
(pps ++ fnl () ++ pidt))
env ~init:(mt ()) in let db_env =
fold_rel_context
(fun env d pps -> let pnat = pr_rel_decl env sigma d in (pps ++ fnl () ++ pnat))
env ~init:(mt ()) in
(sign_env ++ db_env)
let protect f x = try f x with e when (* maybe should be just "not is_interrupted"? *)
CErrors.noncritical e || !Flags.in_debugger ->
str "EXCEPTION: " ++ str (Printexc.to_string e)
let print_kconstr env sigma a =
protect (fun c -> print_constr_env env sigma c) a
end
let vars_of_env env = let s = Environ.ids_of_named_context_val (Environ.named_context_val env) in
Context.Rel.fold_outside
(fun decl s -> match RelDecl.get_name decl with Name id -> Id.Set.add id s | _ -> s)
(rel_context env) ~init:s
let pr_global_env env g = Nametab.pr_global_env (vars_of_env env) g
let evar_suggested_name env sigma evk = letopen Evd in let base_id evk' evi = match evar_ident evk' sigma with
| Some id -> id
| None -> match Evd.evar_source evi with
| _,Evar_kinds.ImplicitArg (c,(n,id),b) -> id
| _,Evar_kinds.VarInstance id -> id
| _,Evar_kinds.QuestionMark {Evar_kinds.qm_name = Name id} -> id
| _,Evar_kinds.GoalEvar -> Id.of_string "Goal"
| _ -> let env = reset_with_named_context (Evd.evar_hyps evi) env in
Namegen.id_of_name_using_hdchar env sigma (Evd.evar_concl evi) Anonymous in let names = Evar.Map.mapi base_id (undefined_map sigma) in let id = Evar.Map.find evk names in let fold evk' id' (seen, n) = if seen then (seen, n) elseif Evar.equal evk evk' then (true, n) elseif Id.equal id id' then (seen, succ n) else (seen, n) in let (_, n) = Evar.Map.fold fold names (false, 0) in if n = 0 then id else Nameops.add_suffix id (string_of_int (pred n))
let pr_existential_key env sigma evk = letopen Evd in match evar_ident evk sigma with
| None ->
str "?" ++ Id.print (evar_suggested_name env sigma evk)
| Some id ->
str "?" ++ Id.print id
let pr_decl env sigma (decl,ok) = letopen NamedDecl in let print_constr = Internal.print_kconstr in match decl with
| LocalAssum ({binder_name=id},_) -> if ok then Id.print id else (str "{" ++ Id.print id ++ str "}")
| LocalDef ({binder_name=id},c,_) -> str (if ok then"("else"{") ++ Id.print id ++ str ":=" ++
print_constr env sigma c ++ str (if ok then")"else"}")
let pr_evar_source env sigma = function
| Evar_kinds.NamedHole id -> Id.print id
| Evar_kinds.QuestionMark _ -> str "underscore"
| Evar_kinds.CasesType false -> str "pattern-matching return predicate"
| Evar_kinds.CasesType true ->
str "subterm of pattern-matching return predicate"
| Evar_kinds.BinderType (Name id) -> str "type of " ++ Id.print id
| Evar_kinds.BinderType Anonymous -> str "type of anonymous binder"
| Evar_kinds.EvarType (ido,evk) -> let pp = match ido with
| Some id -> str "?" ++ Id.print id
| None -> try pr_existential_key env sigma evk with(* defined *) Not_found -> str "an internal placeholder" in
str "type of " ++ pp
| Evar_kinds.ImplicitArg (c,(n,id),b) ->
str "parameter " ++ Id.print id ++ spc () ++ str "of" ++
spc () ++ pr_global_env env c
| Evar_kinds.InternalHole -> str "internal placeholder"
| Evar_kinds.TomatchTypeParameter (ind,n) ->
pr_nth n ++ str " argument of type " ++ pr_global_env env (IndRef ind)
| Evar_kinds.GoalEvar -> str "goal evar"
| Evar_kinds.ImpossibleCase -> str "type of impossible pattern-matching clause"
| Evar_kinds.MatchingVar _ -> str "matching variable"
| Evar_kinds.VarInstance id -> str "instance of " ++ Id.print id
| Evar_kinds.SubEvar (where,evk) ->
(match where with
| None -> str "subterm of "
| Some Evar_kinds.Body -> str "body of "
| Some Evar_kinds.Domain -> str "domain of "
| Some Evar_kinds.Codomain -> str "codomain of ") ++ Evar.print evk
let pr_evar_info (type a) env sigma (evi : a Evd.evar_info) = letopen Evd in let print_constr = Internal.print_kconstr in let phyps = try let decls = matchFilter.repr (evar_filter evi) with
| None -> List.map (fun c -> (c, true)) (evar_context evi)
| Some filter -> List.combine (evar_context evi) filter in
prlist_with_sep spc (pr_decl env sigma) (List.rev decls) with Invalid_argument _ -> str "Ill-formed filtered context"in let pb = match Evd.evar_body evi with
| Evar_empty -> print_constr env sigma (Evd.evar_concl evi)
| Evar_defined c -> str"=> " ++ print_constr env sigma c in let candidates = match Evd.evar_body evi with
| Evar_empty -> beginmatch evar_candidates evi with
| None -> mt ()
| Some l ->
spc () ++ str "{" ++
prlist_with_sep (fun () -> str "|") (print_constr env sigma) l ++ str "}" end
| _ ->
mt () in let src = str "(" ++ pr_evar_source env sigma (snd (Evd.evar_source evi)) ++ str ")"in
hov 2
(str"[" ++ phyps ++ spc () ++ str"|-" ++ spc() ++ pb ++ str"]" ++
candidates ++ spc() ++ src)
let compute_evar_dependency_graph sigma = letopen Evd in (* Compute the map binding ev to the evars whose body depends on ev *) let fold evk (EvarInfo evi) acc = let fold_ev evk' acc = let tab = try Evar.Map.find evk' acc with Not_found -> Evar.Set.empty in
Evar.Map.add evk' (Evar.Set.add evk tab) acc in match evar_body evi with
| Evar_empty -> acc
| Evar_defined c -> Evar.Set.fold fold_ev (evars_of_term sigma c) acc in
Evd.fold fold sigma Evar.Map.empty
let evar_dependency_closure n sigma = letopen Evd in (* Create the DAG of depth [n] representing the recursive dependencies of
undefined evars. *) let graph = compute_evar_dependency_graph sigma in let rec aux n curr accu = if Int.equal n 0 then Evar.Set.union curr accu else let fold evk accu = try let deps = Evar.Map.find evk graph in
Evar.Set.union deps accu with Not_found -> accu in (* Consider only the newly added evars *) let ncurr = Evar.Set.fold fold curr Evar.Set.empty in (* Merge the others *) let accu = Evar.Set.union curr accu in
aux (n - 1) ncurr accu in let undef = Evar.Map.domain (undefined_map sigma) in
aux n undef Evar.Set.empty
let evar_dependency_closure n sigma = letopen Evd in let deps = evar_dependency_closure n sigma in letmap = Evar.Map.bind (fun ev -> find sigma ev) deps in
Evar.Map.bindings map
let pr_evd_level sigma = UState.pr_uctx_level (Evd.ustate sigma)
let pr_evd_qvar sigma = UState.pr_uctx_qvar (Evd.ustate sigma)
let reference_of_level sigma l = UState.qualid_of_level (Evd.ustate sigma) l
let pr_evar_universe_context = UState.pr
let print_env_short env sigma = let print_constr = Internal.print_kconstr in let pr_rel_decl = function
| RelDecl.LocalAssum (n,_) -> Name.print n.binder_name
| RelDecl.LocalDef (n,b,_) -> str "(" ++ Name.print n.binder_name ++ str " := "
++ print_constr env sigma (EConstr.of_constr b) ++ str ")" in let pr_named_decl = NamedDecl.to_rel_decl %> pr_rel_decl in let nc = List.rev (named_context env) in let rc = List.rev (rel_context env) in
str "[" ++ pr_sequence pr_named_decl nc ++ str "]" ++ spc () ++
str "[" ++ pr_sequence pr_rel_decl rc ++ str "]"
let pr_evar_constraints sigma pbs = let pr_evconstr (pbty, env, t1, t2) = let env = (* We currently allow evar instances to refer to anonymous de Bruijn indices, so we protect the error printing code in this case by giving names to every de Bruijn variable in the rel_context of the conversion problem. MS: we should rather stop depending on anonymous variables, they can be used to indicate independency. Also, this depends on a strategy for
naming/renaming. *)
Namegen.make_all_name_different env sigma in
hov 2 (hov 2 (print_env_short env sigma) ++ spc () ++ str "|-" ++ spc () ++
Internal.print_kconstr env sigma t1 ++ spc () ++
str (match pbty with
| Conversion.CONV -> "=="
| Conversion.CUMUL -> "<=") ++
spc () ++ Internal.print_kconstr env sigma t2) in
prlist_with_sep fnl pr_evconstr pbs
let pr_evar_map_gen with_univs pr_evars env sigma = let uvs = Evd.ustate sigma in let (_, conv_pbs) = Evd.extract_all_conv_pbs sigma in let evs = if has_no_evar sigma then mt () else pr_evars sigma ++ fnl () and svs = if with_univs then UState.pr uvs else mt () and cstrs = ifList.is_empty conv_pbs then mt () else
str "CONSTRAINTS:" ++ brk (0, 1) ++
pr_evar_constraints sigma conv_pbs ++ fnl () and typeclasses = let evars = Evd.get_typeclass_evars sigma in if Evar.Set.is_empty evars then mt () else
str "TYPECLASSES:" ++ brk (0, 1) ++
prlist_with_sep spc Evar.print (Evar.Set.elements evars) ++ fnl () and obligations = let evars = Evd.get_obligation_evars sigma in if Evar.Set.is_empty evars then mt () else
str "OBLIGATIONS:" ++ brk (0, 1) ++
prlist_with_sep spc Evar.print (Evar.Set.elements evars) ++ fnl () and shelf =
str "SHELF:" ++ brk (0, 1) ++ Evd.pr_shelf sigma ++ fnl () and future_goals =
str "FUTURE GOALS STACK:" ++ brk (0, 1) ++ Evd.pr_future_goals_stack sigma ++ fnl () in
evs ++ svs ++ cstrs ++ typeclasses ++ obligations ++ shelf ++ future_goals
let pr_evar_list env sigma l = letopen Evd in let pr_alias ev = match is_aliased_evar sigma ev with
| None -> mt ()
| Some ev' -> str " (aliased to " ++ Evar.print ev' ++ str ")" in let pr (ev, EvarInfo evi) =
h (Evar.print ev ++
str "==" ++ pr_evar_info env sigma evi ++
pr_alias ev ++ beginmatch Evd.evar_body evi with
| Evar_empty -> str " {" ++ pr_existential_key env sigma ev ++ str "}"
| Evar_defined _ -> mt () end) in
hv 0 (prlist_with_sep fnl pr l)
let to_list d = letopen Evd in (* Workaround for change in Map.fold behavior in ocaml 3.08.4 *) let l = ref [] in let fold_def evk (EvarInfo evi) () = match Evd.evar_body evi with
| Evar_defined _ -> l := (evk, EvarInfo evi) :: !l
| Evar_empty -> () in let fold_undef evk (EvarInfo evi) () = match Evd.evar_body evi with
| Evar_empty -> l := (evk, EvarInfo evi) :: !l
| Evar_defined _ -> () in
Evd.fold fold_def d ();
Evd.fold fold_undef d ();
!l
let pr_evar_by_depth depth env sigma = match depth with
| None -> (* Print all evars *)
str"EVARS:" ++ brk(0,1) ++ pr_evar_list env sigma (to_list sigma) ++ fnl()
| Some n -> (* Print closure of undefined evars *)
str"UNDEFINED EVARS:"++
(if Int.equal n 0 then mt() else str" (+level "++int n++str" closure):")++
brk(0,1)++
pr_evar_list env sigma (evar_dependency_closure n sigma) ++ fnl()
let pr_evar_by_filter filter env sigma = letopen Evd in let elts = Evd.fold (fun evk evi accu -> (evk, evi) :: accu) sigma [] in let elts = List.rev elts in let is_def (_, EvarInfo evi) = match Evd.evar_body evi with
| Evar_defined _ -> true
| Evar_empty -> false in let (defined, undefined) = List.partition is_def elts in letfilter (evk, evi) = filter evk evi in let defined = List.filterfilter defined in let undefined = List.filterfilter undefined in let prdef = ifList.is_empty defined then mt () else str "DEFINED EVARS:" ++ brk (0, 1) ++
pr_evar_list env sigma defined in let prundef = ifList.is_empty undefined then mt () else str "UNDEFINED EVARS:" ++ brk (0, 1) ++
pr_evar_list env sigma undefined in
prdef ++ prundef
(* [Rel (n+m);...;Rel(n+1)] *) let rel_vect n m = Array.init m (fun i -> mkRel(n+m-i))
let rel_list n m = letopen EConstr in let rec reln l p = if p>m then l else reln (mkRel(n+p)::l) (p+1) in
reln [] 1
let push_rel_assum (x,t) env = letopen RelDecl in letopen EConstr in
push_rel (LocalAssum (x,t)) env
let push_rels_assum assums = letopen RelDecl in
push_rel_context (List.map (fun (x,t) -> LocalAssum (x,t)) assums)
let push_named_rec_types (lna,typarray,_) env = letopen NamedDecl in let ctxt =
Array.map2_i
(fun i na t -> let id = map_annot (function
| Name id -> id
| Anonymous -> anomaly (Pp.str "Fix declarations must be named.")) na in LocalAssum (id, lift i t))
lna typarray in
Array.fold_left
(fun e assum -> push_named assum e) env ctxt
let lookup_rel_id id sign = letopen RelDecl in let rec lookrec n = function
| [] -> raise Not_found
| decl :: l -> if Names.Name.equal (Name id) (get_name decl) then (n, get_value decl, get_type decl) else lookrec (n+1) l in
lookrec 1 sign
let mkProd_or_LetIn = EConstr.mkProd_or_LetIn let mkProd_wo_LetIn = EConstr.mkProd_wo_LetIn
let it_mkProd = EConstr.it_mkProd let it_mkLambda = EConstr.it_mkLambda
let it_mkProd_or_LetIn = EConstr.it_mkProd_or_LetIn let it_mkProd_wo_LetIn = EConstr.it_mkProd_wo_LetIn let it_mkLambda_or_LetIn = Term.it_mkLambda_or_LetIn let it_mkNamedProd_or_LetIn = EConstr.it_mkNamedProd_or_LetIn let it_mkNamedLambda_or_LetIn = EConstr.it_mkNamedLambda_or_LetIn
(* On Constr *) let it_named_context_quantifier f ~init = List.fold_left (fun c d -> f d c) init let it_mkNamedProd_wo_LetIn init = it_named_context_quantifier mkNamedProd_wo_LetIn ~init
let it_mkLambda_or_LetIn_from_no_LetIn c decls = letopen RelDecl in let rec aux k decls c = match decls with
| [] -> c
| LocalDef (na,b,t) :: decls -> mkLetIn (na,b,t,aux (k-1) decls (liftn 1 k c))
| LocalAssum (na,t) :: decls -> mkLambda (na,t,aux (k-1) decls c) in aux (List.length decls) (List.rev decls) c
(* *)
(* strips head casts and flattens head applications *) let rec strip_head_cast sigma c = match EConstr.kind sigma c with
| App (f,cl) -> let rec collapse_rec f cl2 = match EConstr.kind sigma f with
| App (g,cl1) -> collapse_rec g (Array.append cl1 cl2)
| Cast (c,_,_) -> collapse_rec c cl2
| _ -> if Int.equal (Array.length cl2) 0 then f else EConstr.mkApp (f,cl2) in
collapse_rec f cl
| Cast (c,_,_) -> strip_head_cast sigma c
| _ -> c
let rec drop_extra_implicit_args sigma c = match EConstr.kind sigma c with (* Removed trailing extra implicit arguments, what improves compatibility
for constants with recently added maximal implicit arguments *)
| App (f,args) when EConstr.isEvar sigma (Array.last args) -> letopen EConstr in
drop_extra_implicit_args sigma
(mkApp (f,fst (Array.chop (Array.length args - 1) args)))
| _ -> c
(* Get the last arg of an application *) let last_arg sigma c = match EConstr.kind sigma c with
| App (f,cl) -> Array.last cl
| _ -> anomaly (Pp.str "last_arg.")
(* Get the last arg of an application *) let adjust_app_list_size f1 l1 f2 l2 = letopen EConstr in let len1 = List.length l1 and len2 = List.length l2 in if Int.equal len1 len2 then (f1,l1,f2,l2) elseif len1 < len2 then let extras,restl2 = List.chop (len2-len1) l2 in
(f1, l1, applist (f2,extras), restl2) else let extras,restl1 = List.chop (len1-len2) l1 in
(applist (f1,extras), restl1, f2, l2)
let adjust_app_array_size f1 l1 f2 l2 = letopen EConstr in let len1 = Array.length l1 and len2 = Array.length l2 in if Int.equal len1 len2 then (f1,l1,f2,l2) elseif len1 < len2 then let extras,restl2 = Array.chop (len2-len1) l2 in
(f1, l1, mkApp (f2,extras), restl2) else let extras,restl1 = Array.chop (len1-len2) l1 in
(mkApp (f1,extras), restl1, f2, l2)
(* [map_constr_with_binders_left_to_right g f n c] maps [f n] on the immediate subterms of [c]; it carries an extra data [n] (typically a lift index) which is processed by [g] (which typically add 1 to [n]) at each binder traversal; the subterms are processed from left to right according to the usual representation of the constructions (this may matter if [f] does a side-effect); it is not recursive; in fact, the usual representation of the constructions is at the time being almost those of the ML representation (except for
(co-)fixpoint) *)
let fold_rec_types g (lna,typarray,_) e = letopen EConstr in letopen Vars in let ctxt = Array.map2_i (fun i na t -> RelDecl.LocalAssum (na, lift i t)) lna typarray in
Array.fold_left (fun e assum -> g assum e) e ctxt
let map_left2 f a g b = let l = Array.length a in if Int.equal l 0 then [||], [||] elsebegin let r = Array.make l (f a.(0)) in let s = Array.make l (g b.(0)) in
for i = 1 to l - 1 do
r.(i) <- f a.(i);
s.(i) <- g b.(i)
done;
r, s end
let map_constr_with_binders_left_to_right env sigma g f l c = letopen RelDecl in letopen EConstr in match EConstr.kind sigma c with
| (Rel _ | Meta _ | Var _ | Sort _ | Const _ | Ind _
| Construct _ | Int _ | Float _ | String _) -> c
| Cast (b,k,t) -> let b' = f l b in let t' = f l t in if b' == b && t' == t then c else mkCast (b',k,t')
| Prod (na,t,b) -> let t' = f l t in let b' = f (g (LocalAssum (na,t)) l) b in if t' == t && b' == b then c else mkProd (na, t', b')
| Lambda (na,t,b) -> let t' = f l t in let b' = f (g (LocalAssum (na,t)) l) b in if t' == t && b' == b then c else mkLambda (na, t', b')
| LetIn (na,bo,t,b) -> let bo' = f l bo in let t' = f l t in let b' = f (g (LocalDef (na,bo,t)) l) b in if bo' == bo && t' == t && b' == b then c else mkLetIn (na, bo', t', b')
| App (c,[||]) -> assert false
| App (t,al) -> (*Special treatment to be able to recognize partially applied subterms*) let a = al.(Array.length al - 1) in letapp = (mkApp (t, Array.sub al 0 (Array.length al - 1))) in letapp' = f l app in let a' = f l a in ifapp' == app && a' == a then c else mkApp (app', [| a' |])
| Proj (p,r,b) -> let b' = f l b in if b' == b then c else mkProj (p, r, b')
| Evar ev -> let ev' = EConstr.map_existential sigma (fun c -> f l c) ev in if ev' == ev then c else mkEvar ev'
| Case (ci,u,pms,(p,r),iv,b,bl) -> let (ci, _, pms, (p0,_), _, b, bl0) = annotate_case env sigma (ci, u, pms, (p,r), iv, b, bl) in let f_ctx (nas, _ as r) (ctx, c) = let c' = f (List.fold_right g ctx l) c in if c' == c then r else (nas, c') in (* In v8 concrete syntax, predicate is after the term to match! *) let b' = f l b in let pms' = Array.map_left (f l) pms in let p' = f_ctx p p0 in let iv' = map_invert (f l) iv in let bl' = Array.map_left (fun (c, c0) -> f_ctx c c0) (Array.map2 (fun x y -> (x, y)) bl bl0) in if b' == b && pms' == pms && p' == p && iv' == iv && bl' == bl then c else mkCase (ci, u, pms', (p',r), iv', b', bl')
| Fix (ln,(lna,tl,bl as fx)) -> let l' = fold_rec_types g fx l in let (tl', bl') = map_left2 (f l) tl (f l') bl in if Array.for_all2 (==) tl tl' && Array.for_all2 (==) bl bl' then c else mkFix (ln,(lna,tl',bl'))
| CoFix(ln,(lna,tl,bl as fx)) -> let l' = fold_rec_types g fx l in let (tl', bl') = map_left2 (f l) tl (f l') bl in if Array.for_all2 (==) tl tl' && Array.for_all2 (==) bl bl' then c else mkCoFix (ln,(lna,tl',bl'))
| Array(u,t,def,ty) -> let t' = Array.map_left (f l) t in let def' = f l def in let ty' = f l ty in if def' == def && t' == t && ty' == ty then c else mkArray(u,t',def',ty')
(* strong *) let map_constr_with_full_binders env sigma g f l cstr = letopen EConstr in match EConstr.kind sigma cstr with
| (Rel _ | Meta _ | Var _ | Sort _ | Const _ | Ind _
| Construct _ | Int _ | Float _ | String _) -> cstr
| Cast (c,k, t) -> let c' = f l c in let t' = f l t in if c==c' && t==t'then cstr else mkCast (c', k, t')
| Prod (na,t,c) -> let t' = f l t in let c' = f (g (RelDecl.LocalAssum (na, t)) l) c in if t==t' && c==c'then cstr else mkProd (na, t', c')
| Lambda (na,t,c) -> let t' = f l t in let c' = f (g (RelDecl.LocalAssum (na, t)) l) c in if t==t' && c==c'then cstr else mkLambda (na, t', c')
| LetIn (na,b,t,c) -> let b' = f l b in let t' = f l t in let c' = f (g (RelDecl.LocalDef (na, b, t)) l) c in if b==b' && t==t' && c==c' then cstr else mkLetIn (na, b', t', c')
| App (c,al) -> let c' = f l c in let al' = Array.map (f l) al in if c==c' && Array.for_all2 (==) al al'then cstr else mkApp (c', al')
| Proj (p,r,c) -> let c' = f l c in if c' == c then cstr else mkProj (p, r, c')
| Evar ev -> let ev' = EConstr.map_existential sigma (fun c -> f l c) ev in if ev' == ev then cstr else mkEvar ev'
| Case (ci, u, pms, (p,r), iv, c, bl) -> let (ci, _, pms, (p0,_), _, c, bl0) = annotate_case env sigma (ci, u, pms, (p,r), iv, c, bl) in let f_ctx (nas, _ as r) (ctx, c) = let c' = f (List.fold_right g ctx l) c in if c' == c then r else (nas, c') in let pms' = Array.Smart.map (f l) pms in let p' = f_ctx p p0 in let iv' = map_invert (f l) iv in let c' = f l c in let bl' = Array.map2 f_ctx bl bl0 in if pms==pms' && p==p' && iv'==iv && c==c' && Array.for_all2 (==) bl bl' then cstr else
mkCase (ci, u, pms', (p',r), iv', c', bl')
| Fix (ln,(lna,tl,bl as fx)) -> let tl' = Array.map (f l) tl in let l' = fold_rec_types g fx l in let bl' = Array.map (f l') bl in if Array.for_all2 (==) tl tl' && Array.for_all2 (==) bl bl' then cstr else mkFix (ln,(lna,tl',bl'))
| CoFix(ln,(lna,tl,bl as fx)) -> let tl' = Array.map (f l) tl in let l' = fold_rec_types g fx l in let bl' = Array.map (f l') bl in if Array.for_all2 (==) tl tl' && Array.for_all2 (==) bl bl' then cstr else mkCoFix (ln,(lna,tl',bl'))
| Array(u,t,def,ty) -> let t' = Array.Smart.map (f l) t in let def' = f l def in let ty' = f l ty in if def==def' && t == t' && ty==ty' then cstr else mkArray (u,t', def',ty')
(* [fold_constr_with_binders g f n acc c] folds [f n] on the immediate subterms of [c] starting from [acc] and proceeding from left to right according to the usual representation of the constructions as [fold_constr] but it carries an extra data [n] (typically a lift index) which is processed by [g] (which typically add 1 to [n]) at
each binder traversal; it is not recursive *)
let fold_constr_with_full_binders env sigma g f n acc c = letopen EConstr.Vars in letopen Context.Rel.Declaration in match EConstr.kind sigma c with
| Rel _ | Meta _ | Var _ | Sort _ | Const _ | Ind _ | Construct _ | Int _ | Float _ | String _ -> acc
| Cast (c,_, t) -> f n (f n acc c) t
| Prod (na,t,c) -> f (g (LocalAssum (na,t)) n) (f n acc t) c
| Lambda (na,t,c) -> f (g (LocalAssum (na,t)) n) (f n acc t) c
| LetIn (na,b,t,c) -> f (g (LocalDef (na,b,t)) n) (f n (f n acc b) t) c
| App (c,l) -> Array.fold_left (f n) (f n acc c) l
| Proj (_,_,c) -> f n acc c
| Evar ev -> let args = Evd.expand_existential sigma ev in List.fold_left (fun c -> f n c) acc args
| Case (ci, u, pms, p, iv, c, bl) -> let (ci, _, pms, (p,_), _, c, bl) = EConstr.annotate_case env sigma (ci, u, pms, p, iv, c, bl) in let f_ctx acc (ctx, c) = f (List.fold_right g ctx n) acc c in
Array.fold_left f_ctx (f n (fold_invert (f n) (f_ctx (Array.fold_left (f n) acc pms) p) iv) c) bl
| Fix (_,(lna,tl,bl)) -> let n' = CArray.fold_left2_i (fun i c n t -> g (LocalAssum (n,lift i t)) c) n lna tl in let fd = Array.map2 (fun t b -> (t,b)) tl bl in
Array.fold_left (fun acc (t,b) -> f n' (f n acc t) b) acc fd
| CoFix (_,(lna,tl,bl)) -> let n' = CArray.fold_left2_i (fun i c n t -> g (LocalAssum (n,lift i t)) c) n lna tl in let fd = Array.map2 (fun t b -> (t,b)) tl bl in
Array.fold_left (fun acc (t,b) -> f n' (f n acc t) b) acc fd
| Array(_u,t,def,ty) -> f n (f n (Array.fold_left (f n) acc t) def) ty
let occur_meta sigma c = let rec occrec h c = let h, knd = EConstr.Expand.kind sigma h c in match knd with
| Meta _ -> raise Occur
| Evar (evk, args) -> let evi = Evd.find_undefined sigma evk in let args = EConstr.Expand.expand_instance ~skip:true evi h args in
SList.Skip.iter (fun c -> occrec h c) args
| _ -> EConstr.Expand.iter sigma occrec h knd in let h, c = EConstr.Expand.make c in try occrec h c; falsewith Occur -> true
let occur_existential sigma c = let rec occrec h c = let h, knd = EConstr.Expand.kind sigma h c in match knd with
| Evar _ -> raise Occur
| _ -> EConstr.Expand.iter sigma occrec h knd in let h, c = EConstr.Expand.make c in try occrec h c; falsewith Occur -> true
let occur_meta_or_existential sigma c = let rec occrec h c = let h, knd = EConstr.Expand.kind sigma h c in match knd with
| Evar _ -> raise Occur
| Meta _ -> raise Occur
| _ -> EConstr.Expand.iter sigma occrec h knd in let h, c = EConstr.Expand.make c in try occrec h c; falsewith Occur -> true
let occur_metavariable sigma m c = let rec occrec c = match EConstr.kind sigma c with
| Meta m' -> if Int.equal m m'thenraise Occur
| Evar (_, args) -> SList.Skip.iter occrec args
| _ -> EConstr.iter sigma occrec c in try occrec c; falsewith Occur -> true
let occur_evar sigma n c = let rec occur_rec c = match EConstr.kind sigma c with
| Evar (sp, args) -> if Evar.equal sp n thenraise Occur else SList.Skip.iter occur_rec args
| _ -> EConstr.iter sigma occur_rec c in try occur_rec c; falsewith Occur -> true
let occur_in_global env id constr = let vars = vars_of_global env constr in
Id.Set.mem id vars
let occur_var env sigma id c = let rec occur_rec c = match EConstr.destRef sigma c with
| gr, _ -> if occur_in_global env id gr thenraise Occur
| exception DestKO -> EConstr.iter sigma occur_rec c in try occur_rec c; falsewith Occur -> true
let occur_vars env sigma ids c = let rec occur_rec c = match EConstr.destRef sigma c with
| gr, _ -> let vars = vars_of_global env gr in ifnot (Id.Set.is_empty (Id.Set.inter ids vars)) thenraise Occur
| exception DestKO -> EConstr.iter sigma occur_rec c in try occur_rec c; falsewith Occur -> true
exception OccurInGlobal of GlobRef.t
let occur_var_indirectly env sigma id c = let var = GlobRef.VarRef id in let rec occur_rec c = match EConstr.destRef sigma c with
| gr, _ -> ifnot (QGlobRef.equal env gr var) && occur_in_global env id gr thenraise (OccurInGlobal gr)
| exception DestKO -> EConstr.iter sigma occur_rec c in try occur_rec c; None with OccurInGlobal gr -> Some gr
let local_occur_var sigma id c = let rec occur c = match EConstr.kind sigma c with
| Var id' -> if Id.equal id id'thenraise Occur
| _ -> EConstr.iter sigma occur c in try occur c; falsewith Occur -> true
(* returns the list of free debruijn indices in a term *)
let free_rels sigma m = let rec frec depth acc c = match EConstr.kind sigma c with
| Rel n -> if n >= depth then Int.Set.add (n-depth+1) acc else acc
| Evar (_, args) -> SList.Skip.fold (fun acc c -> frec depth acc c) acc args
| _ -> EConstr.fold_with_binders sigma succ frec depth acc c in
frec 1 Int.Set.empty m
let free_rels_and_unqualified_refs sigma t = let rec aux k (gseen, vseen, ids as accu) t = match EConstr.kind sigma t with
| Const _ | Ind _ | Construct _ | Var _ -> let g, _ = EConstr.destRef sigma t in ifnot (GlobRef.Set_env.mem g gseen) thenbegin try let gseen = GlobRef.Set_env.add g gseen in let short = Nametab.shortest_qualid_of_global Id.Set.empty g in let dir, id = Libnames.repr_qualid short in let ids = if DirPath.is_empty dir then Id.Set.add id ids else ids in
(gseen, vseen, ids) with Not_found when !Flags.in_debugger || !Flags.in_ml_toplevel ->
accu endelse
accu
| Rel p -> if p > k && not (Int.Set.mem (p - k) vseen) then let vseen = Int.Set.add (p - k) vseen in
(gseen, vseen, ids) else
accu
| _ ->
EConstr.fold_with_binders sigma succ aux k accu t in let accu = (GlobRef.Set_env.empty, Int.Set.empty, Id.Set.empty) in let (_, rels, ids) = aux 0 accu t in
rels, ids
(* collects all metavar occurrences, in left-to-right order, preserving
* repetitions and all. *)
let collect_metas sigma c = let rec collrec acc c = match EConstr.kind sigma c with
| Meta mv -> List.add_set Int.equal mv acc
| Evar (_, args) -> SList.Skip.fold collrec acc args
| _ -> EConstr.fold sigma collrec acc c in List.rev (collrec [] c)
(* collects all vars; warning: this is only visible vars, not dependencies in
all section variables; for the latter, use global_vars_set *) let collect_vars sigma c = let rec aux vars c = match EConstr.kind sigma c with
| Var id -> Id.Set.add id vars
| _ -> EConstr.fold sigma aux vars c in
aux Id.Set.empty c
(* Tests whether [m] is a subterm of [t]:
[m] is appropriately lifted through abstractions of [t] *)
let dependent_main noevar sigma m t = letopen EConstr in let eqc x y = eq_constr_nounivs sigma x y in let rec deprec m t = if eqc m t then raise Occur else match EConstr.kind sigma m, EConstr.kind sigma t with
| App (fm,lm), App (ft,lt) when Array.length lm < Array.length lt ->
deprec m (mkApp (ft,Array.sub lt 0 (Array.length lm)));
Array.Fun1.iter deprec m
(Array.sub lt
(Array.length lm) ((Array.length lt) - (Array.length lm)))
| _, Cast (c,_,_) when noevar && isMeta sigma c -> ()
| _, Evar _ when noevar -> ()
| _ -> EConstr.iter_with_binders sigma (fun c -> Vars.lift 1 c) deprec m t in try deprec m t; falsewith Occur -> true
let dependent sigma c t = dependent_main false sigma c t let dependent_no_evar sigma c t = dependent_main true sigma c t
let dependent_in_decl sigma a decl = letopen NamedDecl in match decl with
| LocalAssum (_,t) -> dependent sigma a t
| LocalDef (_, body, t) -> dependent sigma a body || dependent sigma a t
let count_occurrences sigma m t = letopen EConstr in let n = ref 0 in let rec countrec m t = if EConstr.eq_constr sigma m t then
incr n else match EConstr.kind sigma m, EConstr.kind sigma t with
| App (fm,lm), App (ft,lt) when Array.length lm < Array.length lt ->
countrec m (mkApp (ft,Array.sub lt 0 (Array.length lm)));
Array.iter (countrec m)
(Array.sub lt
(Array.length lm) ((Array.length lt) - (Array.length lm)))
| _, Cast (c,_,_) when isMeta sigma c -> ()
| _, Evar _ -> ()
| _ -> EConstr.iter_with_binders sigma (Vars.lift 1) countrec m t in
countrec m t;
!n
type meta_value_map = (metavariable * constr) list
let isMetaOf sigma mv c = match EConstr.kind sigma c with Meta mv' -> Int.equal mv mv' | _ -> false
let rec subst_meta bl c = match kind c with
| Meta i -> (try Int.List.assoc i bl with Not_found -> c)
| _ -> Constr.map (subst_meta bl) c
let rec strip_outer_cast sigma c = match EConstr.kind sigma c with
| Cast (c,_,_) -> strip_outer_cast sigma c
| _ -> c
(* First utilities for avoiding telescope computation for subst_term *)
let prefix_application sigma eq_fun k l1 t = letopen EConstr in if 0 < l1 thenmatch EConstr.kind sigma t with
| App (f2,cl2) -> let l2 = Array.length cl2 in if l1 <= l2
&& eq_fun sigma k (mkApp (f2, Array.sub cl2 0 l1)) then
Some (Array.sub cl2 l1 (l2 - l1)) else
None
| _ -> None else None
let eq_upto_lift cache c sigma k t = let c = try Int.Map.find k !cache with Not_found -> let c = EConstr.Vars.lift k c in let () = cache := Int.Map.add k c !cache in
c in
EConstr.eq_constr sigma c t
(* Recognizing occurrences of a given subterm in a term : [replace_term c1 c2 t] substitutes [c2] for all occurrences of
term [c1] in a term [t]; works if [c1] and [c2] have rels *)
let replace_term_gen sigma eq_fun ar by_c in_t = let rec substrec k t = match prefix_application sigma eq_fun k ar t with
| Some args -> EConstr.mkApp (EConstr.Vars.lift k by_c, args)
| None ->
(if eq_fun sigma k t then (EConstr.Vars.lift k by_c) else
EConstr.map_with_binders sigma succ substrec k t) in
substrec 0 in_t
let replace_term sigma c byc t = let cache = ref Int.Map.empty in let ar = Array.length (snd (EConstr.decompose_app sigma c)) in let eq sigma k t = eq_upto_lift cache c sigma k t in
replace_term_gen sigma eq ar byc t
let subst_term sigma c t = replace_term sigma c (EConstr.mkRel 1) t
let add_vname vars = function
Name id -> Id.Set.add id vars
| _ -> vars
(*************************) (* Names environments *) (*************************) type names_context = Name.t list let add_name n nl = n::nl let lookup_name_of_rel p names = tryList.nth names (p-1) with Invalid_argument _ | Failure _ -> raise Not_found let lookup_rel_of_name id names = let rec lookrec n = function
| Anonymous :: l -> lookrec (n+1) l
| (Name id') :: l -> if Id.equal id' id then n else lookrec (n+1) l
| [] -> raise Not_found in
lookrec 1 names let empty_names_context = []
let ids_of_rel_context sign =
Context.Rel.fold_outside
(fun decl l -> match RelDecl.get_name decl with Name id -> id::l | Anonymous -> l)
sign ~init:[]
let names_of_rel_context env = List.map RelDecl.get_name (rel_context env)
let is_section_variable env id = trylet _ = Environ.lookup_named id env intrue with Not_found -> false
let global_of_constr sigma c = letopen GlobRef in match EConstr.kind sigma c with
| Const (c, u) -> ConstRef c, u
| Ind (i, u) -> IndRef i, u
| Construct (c, u) -> ConstructRef c, u
| Var id -> VarRef id, EConstr.EInstance.empty
| _ -> raise Not_found
let is_global = EConstr.isRefX
let isGlobalRef = EConstr.isRef
let is_template_polymorphic_ref env sigma f = match EConstr.kind sigma f with
| Ind (ind, u) | Construct ((ind, _), u) -> ifnot (EConstr.EInstance.is_empty u) thenfalse else Environ.template_polymorphic_ind ind env
| _ -> false
let is_template_polymorphic_ind env sigma f = match EConstr.kind sigma f with
| Ind (ind, u) -> ifnot (EConstr.EInstance.is_empty u) thenfalse else Environ.template_polymorphic_ind ind env
| _ -> false
let base_sort_cmp pb s0 s1 = match (s0,s1) with
| SProp, SProp | Prop, Prop | Set, Set | Type _, Type _ -> true
| QSort (q1, _), QSort (q2, _) -> Sorts.QVar.equal q1 q2
| QSort _, _ | _, QSort _ -> false
| SProp, _ | _, SProp -> false
| Prop, Set | Prop, Type _ | Set, Type _ -> pb == Conversion.CUMUL
| Set, Prop | Type _, Prop | Type _, Set -> false
let rec is_Prop sigma c = match EConstr.kind sigma c with
| Sort u -> beginmatch EConstr.ESorts.kind sigma u with
| Prop -> true
| _ -> false end
| Cast (c,_,_) -> is_Prop sigma c
| _ -> false
let rec is_Set sigma c = match EConstr.kind sigma c with
| Sort u -> beginmatch EConstr.ESorts.kind sigma u with
| Set -> true
| _ -> false end
| Cast (c,_,_) -> is_Set sigma c
| _ -> false
let rec is_Type sigma c = match EConstr.kind sigma c with
| Sort u -> beginmatch EConstr.ESorts.kind sigma u with
| Type _ -> true
| _ -> false end
| Cast (c,_,_) -> is_Type sigma c
| _ -> false
(* eq_constr extended with universe erasure *) let compare_constr_univ env sigma f cv_pb t1 t2 = letopen EConstr in match EConstr.kind sigma t1, EConstr.kind sigma t2 with
Sort s1, Sort s2 -> base_sort_cmp cv_pb (ESorts.kind sigma s1) (ESorts.kind sigma s2)
| Prod (_,t1,c1), Prod (_,t2,c2) ->
f Conversion.CONV t1 t2 && f cv_pb c1 c2
| Const (c, u), Const (c', u') -> QConstant.equal env c c'
| Ind (i, _), Ind (i', _) -> QInd.equal env i i'
| Construct (i, _), Construct (i', _) -> QConstruct.equal env i i'
| _ -> EConstr.compare_constr sigma (fun t1 t2 -> f Conversion.CONV t1 t2) t1 t2
let constr_cmp env sigma cv_pb t1 t2 = let rec compare cv_pb t1 t2 = compare_constr_univ env sigma compare cv_pb t1 t2 in
compare cv_pb t1 t2
(* (nb_lam [na1:T1]...[nan:Tan]c) where c is not an abstraction
* gives n (casts are ignored) *) let nb_lam sigma c = let rec nbrec n c = match EConstr.kind sigma c with
| Lambda (_,_,c) -> nbrec (n+1) c
| Cast (c,_,_) -> nbrec n c
| _ -> n in
nbrec 0 c
(* similar to nb_lam, but gives the number of products instead *) let nb_prod sigma c = let rec nbrec n c = match EConstr.kind sigma c with
| Prod (_,_,c) -> nbrec (n+1) c
| Cast (c,_,_) -> nbrec n c
| _ -> n in
nbrec 0 c
let nb_prod_modulo_zeta sigma x = let rec count n c = match EConstr.kind sigma c with
Prod(_,_,t) -> count (n+1) t
| LetIn(_,a,_,t) -> count n (EConstr.Vars.subst1 a t)
| Cast(c,_,_) -> count n c
| _ -> n in count 0 x
(* We reduce a series of head eta-redex or nothing at all *) (* [x1:c1;...;xn:cn]@(f;a1...an;x1;...;xn) --> @(f;a1...an) *) (* Remplace 2 earlier buggish versions *)
let rec eta_reduce_head sigma c = letopen EConstr in letopen Vars in match EConstr.kind sigma c with
| Lambda (_,c1,c') ->
(match EConstr.kind sigma (eta_reduce_head sigma c') with
| App (f,cl) -> let lastn = (Array.length cl) - 1 in if lastn < 0 then anomaly (Pp.str "application without arguments.") else
(match EConstr.kind sigma cl.(lastn) with
| Rel 1 -> let c' = if Int.equal lastn 0 then f else mkApp (f, Array.sub cl 0 lastn) in if noccurn sigma 1 c' then lift (-1) c' else c
| _ -> c)
| _ -> c)
| _ -> c
(* iterator on rel context *) let process_rel_context f env = let sign = named_context_val env in let rels = EConstr.rel_context env in let env0 = reset_with_named_context sign env in
Context.Rel.fold_outside f rels ~init:env0
let assums_of_rel_context sign =
Context.Rel.fold_outside
(fun decl l -> match decl with
| RelDecl.LocalDef _ -> l
| RelDecl.LocalAssum (na,t) -> (na, t)::l)
sign ~init:[]
let map_rel_context_in_env f env sign = let rec aux env acc = function
| d::sign ->
aux (push_rel d env) (RelDecl.map_constr (f env) d :: acc) sign
| [] ->
acc in
aux env [] (List.rev sign)
let map_rel_context_with_binders = Context.Rel.map_with_binders let substl_rel_context = Vars.substl_rel_context let lift_rel_context = Vars.lift_rel_context let smash_rel_context = Vars.smash_rel_context
let fold_named_context_both_sides f l ~init = List.fold_right_and_left f l init
let mem_named_context_val id ctxt = try ignore(Environ.lookup_named_ctxt id ctxt); truewith Not_found -> false
let clear_named_body id env = letopen NamedDecl in let aux _ = function
| LocalDef (id',c,t) when Id.equal id id'.binder_name -> push_named (LocalAssum (id',t))
| d -> push_named d in
fold_named_context aux env ~init:(reset_context env)
let global_vars_set env sigma constr = let rec filtrec acc c = match EConstr.destRef sigma c with
| gr, _ -> Id.Set.union (vars_of_global env gr) acc
| exception DestKO -> EConstr.fold sigma filtrec acc c in
filtrec Id.Set.empty constr
let global_vars_set_of_decl env sigma = function
| NamedDecl.LocalAssum (_,t) -> global_vars_set env sigma t
| NamedDecl.LocalDef (_,c,t) ->
Id.Set.union (global_vars_set env sigma t)
(global_vars_set env sigma c)
let dependency_closure env sigma sign hyps = if Id.Set.is_empty hyps then [] else let (_,lh) =
Context.Named.fold_inside
(fun (hs,hl) d -> let x = NamedDecl.get_id d in if Id.Set.mem x hs then
(Id.Set.union (global_vars_set_of_decl env sigma d) (Id.Set.remove x hs),
x::hl) else (hs,hl))
~init:(hyps,[])
sign in List.rev lh
let global_app_of_constr sigma c = letopen GlobRef in match EConstr.kind sigma c with
| Const (c, u) -> (ConstRef c, u), None
| Ind (i, u) -> (IndRef i, u), None
| Construct (c, u) -> (ConstructRef c, u), None
| Var id -> (VarRef id, EConstr.EInstance.empty), None
| Proj (p, _, c) -> (ConstRef (Projection.constant p), EConstr.EInstance.empty), Some c
| _ -> raise Not_found
let prod_applist sigma c l = letopen EConstr in let rec app subst c l = match EConstr.kind sigma c, l with
| Prod(_,_,c), arg::l -> app (arg::subst) c l
| _, [] -> Vars.substl subst c
| _ -> anomaly (Pp.str "Not enough prod's.") in app [] c l
let prod_applist_decls sigma n c l = letopen EConstr in let rec app n subst c l = if Int.equal n 0 then if l == [] then Vars.substl subst c else anomaly (Pp.str "Not enough arguments.") elsematch EConstr.kind sigma c, l with
| Prod(_,_,c), arg::l -> app (n-1) (arg::subst) c l
| LetIn(_,b,_,c), _ -> app (n-1) (Vars.substl subst b::subst) c l
| _ -> anomaly (Pp.str "Not enough prod/let's.") in app n [] c l
(* Cut a context ctx in 2 parts (ctx1,ctx2) with ctx1 containing k non let-in
variables skips let-in's; let-in's in the middle are put in ctx2 *) let context_chop k ctx = let rec chop_aux acc = function
| (0, l2) -> (List.rev acc, l2)
| (n, (RelDecl.LocalDef _ as h)::t) -> chop_aux (h::acc) (n, t)
| (n, (h::t)) -> chop_aux (h::acc) (pred n, t)
| (_, []) -> anomaly (Pp.str "context_chop.") in chop_aux [] (k,ctx)
(* Do not skip let-in's *) let env_rel_context_chop k env = letopen EConstr in let rels = rel_context env in let ctx1,ctx2 = List.chop k rels in
push_rel_context ctx2 (reset_with_named_context (named_context_val env) env),
ctx1
¤ Dauer der Verarbeitung: 0.14 Sekunden
(vorverarbeitet)
¤
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung ist noch experimentell.