diff options
author | Stephane Glondu <steph@glondu.net> | 2010-12-24 11:53:29 +0100 |
---|---|---|
committer | Stephane Glondu <steph@glondu.net> | 2010-12-24 11:53:29 +0100 |
commit | 6b691bbd2101fd39395c0d2135fd7c06a8915e14 (patch) | |
tree | b04b45d1a6f42d19b1428c522d647afbad2f9b83 /kernel/univ.ml | |
parent | 3e96002677226c0cdaa8f355938a76cfb37a722a (diff) |
Imported Upstream version 8.3pl1upstream/8.3pl1
Diffstat (limited to 'kernel/univ.ml')
-rw-r--r-- | kernel/univ.ml | 114 |
1 files changed, 53 insertions, 61 deletions
diff --git a/kernel/univ.ml b/kernel/univ.ml index 77c14b10..9d93b16f 100644 --- a/kernel/univ.ml +++ b/kernel/univ.ml @@ -6,7 +6,7 @@ (* * GNU Lesser General Public License Version 2.1 *) (************************************************************************) -(* $Id: univ.ml 13323 2010-07-24 15:57:30Z herbelin $ *) +(* $Id: univ.ml 13735 2010-12-21 18:21:58Z letouzey $ *) (* Initial Caml version originates from CoC 4.8 [Dec 1988] *) (* Extension with algebraic universes by HH [Sep 2001] *) @@ -115,15 +115,16 @@ let terminal u = {univ=u; lt=[]; le=[]} (* A universe_level is either an alias for another one, or a canonical one, for which we know the universes that are above *) + type univ_entry = Canonical of canonical_arc - | Equiv of universe_level * universe_level + | Equiv of universe_level type universes = univ_entry UniverseLMap.t let enter_equiv_arc u v g = - UniverseLMap.add u (Equiv(u,v)) g + UniverseLMap.add u (Equiv v) g let enter_arc ca g = UniverseLMap.add ca.univ (Canonical ca) g @@ -173,34 +174,13 @@ let repr g u = (str"Universe " ++ pr_uni_level u ++ str" undefined") in match a with - | Equiv(_,v) -> repr_rec v + | Equiv v -> repr_rec v | Canonical arc -> arc in repr_rec u let can g = List.map (repr g) -(* transitive closure : we follow the Less links *) - -(* collect : canonical_arc -> canonical_arc list * canonical_arc list *) -(* collect u = (V,W) iff V={v canonical | u<v} W={w canonical | u<=w}-V *) -(* i.e. collect does the transitive upward closure of what is known about u *) -let collect g arcu = - let rec coll_rec lt le = function - | [],[] -> (lt, list_subtractq le lt) - | arcv::lt', le' -> - if List.memq arcv lt then - coll_rec lt le (lt',le') - else - coll_rec (arcv::lt) le ((can g (arcv.lt@arcv.le))@lt',le') - | [], arcw::le' -> - if (List.memq arcw lt) or (List.memq arcw le) then - coll_rec lt le ([],le') - else - coll_rec lt (arcw::le) (can g arcw.lt, (can g arcw.le)@le') - in - coll_rec [] [] ([],[arcu]) - (* reprleq : canonical_arc -> canonical_arc list *) (* All canonical arcv such that arcu<=arcv with arcv#arcu *) let reprleq g arcu = @@ -252,20 +232,50 @@ let between g u arcv = type order = EQ | LT | LE | NLE -(* compare : universe_level -> universe_level -> order *) +(** [compare_neq] : is [arcv] in the transitive upward closure of [arcu] ? + + We try to avoid visiting unneeded parts of this transitive closure, + by stopping as soon as [arcv] is encountered. During the recursive + traversal, [lt_done] and [le_done] are universes we have already + visited, they do not contain [arcv]. The 3rd arg is + [(lt_todo,le_todo)], two lists of universes not yet considered, + known to be above [arcu], strictly or not. + + We use depth-first search, but the presence of [arcv] in [new_lt] + is checked as soon as possible : this seems to be slightly faster + on a test. +*) + +let compare_neq g arcu arcv = + let rec cmp lt_done le_done = function + | [],[] -> NLE + | arc::lt_todo, le_todo -> + if List.memq arc lt_done then + cmp lt_done le_done (lt_todo,le_todo) + else + let lt_new = can g (arc.lt@arc.le) in + if List.memq arcv lt_new then LT + else cmp (arc::lt_done) le_done (lt_new@lt_todo,le_todo) + | [], arc::le_todo -> + if arc == arcv then LE + (* No need to continue inspecting universes above arc: + if arcv is strictly above arc, then we would have a cycle *) + else + if (List.memq arc lt_done) || (List.memq arc le_done) then + cmp lt_done le_done ([],le_todo) + else + let lt_new = can g arc.lt in + if List.memq arcv lt_new then LT + else + let le_new = can g arc.le in + cmp lt_done (arc::le_done) (lt_new, le_new@le_todo) + in + cmp [] [] ([],[arcu]) + let compare g u v = let arcu = repr g u and arcv = repr g v in - if arcu==arcv then - EQ - else - let (lt,leq) = collect g arcu in - if List.memq arcv lt then - LT - else if List.memq arcv leq then - LE - else - NLE + if arcu == arcv then EQ else compare_neq g arcu arcv (* Invariants : compare(u,v) = EQ <=> compare(v,u) = EQ compare(u,v) = LT or LE => compare(v,u) = NLE @@ -299,9 +309,6 @@ let rec check_eq g u v = compare_list (compare_eq g) ult vlt | _ -> anomaly "check_eq" (* not complete! (Atom(u) = Max([u],[]) *) -let check_eq g u v = - check_eq g u v - let compare_greater g strict u v = let g = declare_univ u g in let g = declare_univ v g in @@ -429,21 +436,6 @@ let enforce_univ_lt u v g = | NLE -> setlt g u v | _ -> error_inconsistency Lt u v) -(* -let enforce_univ_relation g = function - | Equiv (u,v) -> enforce_univ_eq u v g - | Canonical {univ=u; lt=lt; le=le} -> - let g' = List.fold_right (enforce_univ_lt u) lt g in - List.fold_right (enforce_univ_leq u) le g' -*) - -(* Merging 2 universe graphs *) -(* -let merge_universes sp u1 u2 = - UniverseLMap.fold (fun _ a g -> enforce_univ_relation g a) u1 u2 -*) - - (* Constraints and sets of consrtaints. *) type constraint_type = Lt | Leq | Eq @@ -578,21 +570,21 @@ let num_edges g = UniverseLMap.fold (fun _ a n -> n + (reln_len a)) g 0 let pr_arc = function - | Canonical {univ=u; lt=[]; le=[]} -> + | _, Canonical {univ=u; lt=[]; le=[]} -> mt () - | Canonical {univ=u; lt=lt; le=le} -> + | _, Canonical {univ=u; lt=lt; le=le} -> pr_uni_level u ++ str " " ++ v 0 (prlist_with_sep pr_spc (fun v -> str "< " ++ pr_uni_level v) lt ++ (if lt <> [] & le <> [] then spc () else mt()) ++ prlist_with_sep pr_spc (fun v -> str "<= " ++ pr_uni_level v) le) ++ fnl () - | Equiv (u,v) -> + | u, Equiv v -> pr_uni_level u ++ str " = " ++ pr_uni_level v ++ fnl () let pr_universes g = - let graph = UniverseLMap.fold (fun k a l -> (k,a)::l) g [] in - prlist (function (_,a) -> pr_arc a) graph + let graph = UniverseLMap.fold (fun u a l -> (u,a)::l) g [] in + prlist pr_arc graph let pr_constraints c = Constraint.fold (fun (u1,op,u2) pp_std -> @@ -606,7 +598,7 @@ let pr_constraints c = (* Dumping constraints to a file *) let dump_universes output g = - let dump_arc _ = function + let dump_arc u = function | Canonical {univ=u; lt=lt; le=le} -> let u_str = string_of_univ_level u in List.iter @@ -619,7 +611,7 @@ let dump_universes output g = Printf.fprintf output "%s <= %s ;\n" u_str (string_of_univ_level v)) le - | Equiv (u,v) -> + | Equiv v -> Printf.fprintf output "%s = %s ;\n" (string_of_univ_level u) (string_of_univ_level v) in |