summaryrefslogtreecommitdiff
path: root/kernel/univ.ml
diff options
context:
space:
mode:
authorGravatar Stephane Glondu <steph@glondu.net>2010-12-24 11:53:29 +0100
committerGravatar Stephane Glondu <steph@glondu.net>2010-12-24 11:53:29 +0100
commit6b691bbd2101fd39395c0d2135fd7c06a8915e14 (patch)
treeb04b45d1a6f42d19b1428c522d647afbad2f9b83 /kernel/univ.ml
parent3e96002677226c0cdaa8f355938a76cfb37a722a (diff)
Imported Upstream version 8.3pl1upstream/8.3pl1
Diffstat (limited to 'kernel/univ.ml')
-rw-r--r--kernel/univ.ml114
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