(* $Id: dag2html.ml,v 5.0 2005-12-13 11:51:26 ddr Exp $ *) open Gwdb type 'a dag = { mutable dag : 'a node array } and 'a node = { mutable pare : idag list; valu : 'a; mutable chil : idag list } and idag external int_of_idag : idag -> int = "%identity" external idag_of_int : int -> idag = "%identity" type 'a table = { mutable table : 'a data array array } and 'a data = { mutable elem : 'a elem; mutable span : span_id } and 'a elem = Elem of 'a | Ghost of ghost_id | Nothing and span_id and ghost_id external span_id_of_int : int -> span_id = "%identity" external ghost_id_of_int : int -> ghost_id = "%identity" let new_span_id = let i = ref 0 in fun () -> incr i; span_id_of_int !i let new_ghost_id = let i = ref 0 in fun () -> incr i; ghost_id_of_int !i (* creating the html table structure *) type align = LeftA | CenterA | RightA type 'a table_data = | TDitem of iper * 'a * Adef.safe_string | TDtext of iper * Adef.safe_string | TDhr of align | TDbar of Adef.escaped_string option | TDnothing type 'a html_table_line = (int * align * 'a table_data) array type 'a html_table = 'a html_table_line array let html_table_struct indi_ip indi_txt vbar_txt phony d t = let phony = function | Elem e -> phony d.dag.(int_of_idag e) | Ghost _ -> false | Nothing -> true in let elem_txt = function | Elem e -> TDitem ( indi_ip d.dag.(int_of_idag e), indi_txt d.dag.(int_of_idag e), Adef.safe "" ) | Ghost _ -> TDbar None | Nothing -> TDnothing in let bar_txt first_vbar = function | Elem e -> let b = if first_vbar then Some (vbar_txt d.dag.(int_of_idag e)) else None in TDbar b | Ghost _ -> TDbar None | Nothing -> TDnothing in let all_empty i = let rec loop j = if j = Array.length t.table.(i) then true else match t.table.(i).(j).elem with | Nothing -> loop (j + 1) | (Elem _ | Ghost _) as e -> if phony e then loop (j + 1) else false in loop 0 in let line_elem_txt i = let les = let rec loop les j = if j = Array.length t.table.(i) then les else let x = t.table.(i).(j) in let next_j = let rec loop j = if j = Array.length t.table.(i) then j else if t.table.(i).(j) = x then loop (j + 1) else j in loop (j + 1) in let colspan = 3 * (next_j - j) in let les = (1, LeftA, TDnothing) :: les in let les = let td = if t.table.(i).(j).elem = Nothing then TDnothing else elem_txt t.table.(i).(j).elem in (colspan - 2, CenterA, td) :: les in let les = (1, LeftA, TDnothing) :: les in loop les next_j in loop [] 0 in Array.of_list (List.rev les) in let vbars_txt k i = let les = let rec loop les j = if j = Array.length t.table.(i) then les else let x = t.table.(i).(j) in let next_j = let rec loop j = if j = Array.length t.table.(i) then j else if t.table.(i).(j) = x then loop (j + 1) else j in loop (j + 1) in let colspan = 3 * (next_j - j) in let les = (1, LeftA, TDnothing) :: les in let les = let td = if (k > 0 && t.table.(k - 1).(j).elem = Nothing) || t.table.(k).(j).elem = Nothing then TDnothing else if phony t.table.(i).(j).elem then TDnothing else bar_txt (k <> i) t.table.(i).(j).elem in (colspan - 2, CenterA, td) :: les in let les = (1, LeftA, TDnothing) :: les in loop les next_j in loop [] 0 in Array.of_list (List.rev les) in let alone_bar_txt i = let les = let rec loop les j = if j = Array.length t.table.(i) then les else let next_j = let x = t.table.(i).(j).span in let rec loop j = if j = Array.length t.table.(i) then j else if t.table.(i).(j).span = x then loop (j + 1) else j in loop (j + 1) in let colspan = (3 * (next_j - j)) - 2 in let les = (1, LeftA, TDnothing) :: les in let les = if t.table.(i).(j).elem = Nothing || t.table.(i + 1).(j).elem = Nothing then (colspan, LeftA, TDnothing) :: les else let td = let all_ph = let rec loop j = if j = next_j then true else if phony t.table.(i + 1).(j).elem then loop (j + 1) else false in loop j in if all_ph then TDnothing else TDbar None in (colspan, CenterA, td) :: les in let les = (1, LeftA, TDnothing) :: les in loop les next_j in loop [] 0 in Array.of_list (List.rev les) in let exist_several_branches i k = let rec loop j = if j = Array.length t.table.(i) then false else let x = t.table.(i).(j).span in let e = t.table.(k).(j).elem in let rec loop1 j = if j = Array.length t.table.(i) then false else if t.table.(i).(j).elem = Nothing then loop j else if t.table.(i).(j).span <> x then loop j else if t.table.(k).(j).elem <> e then true else loop1 (j + 1) in loop1 (j + 1) in loop 0 in let hbars_txt i k = let les = let rec loop les j = if j = Array.length t.table.(i) then les else let next_j = let e = t.table.(i).(j).elem in let x = t.table.(i).(j).span in let rec loop j = if j = Array.length t.table.(i) then j else if e = Nothing && t.table.(i).(j).elem = Nothing then loop (j + 1) else if t.table.(i).(j).span = x then loop (j + 1) else j in loop (j + 1) in let rec loop1 les l = if l = next_j then loop les next_j else let next_l = let y = t.table.(k).(l) in match y.elem with | Elem _ | Ghost _ -> let rec loop l = if l = Array.length t.table.(i) then l else if t.table.(k).(l) = y then loop (l + 1) else l in loop (l + 1) | Nothing -> l + 1 in if next_l > next_j then ( Printf.eprintf "assert false i %d k %d l %d next_l %d next_j %d\n" i k l next_l next_j; flush stderr); let next_l = min next_l next_j in let colspan = (3 * (next_l - l)) - 2 in let les = match (t.table.(i).(l).elem, t.table.(i + 1).(l).elem) with | Nothing, _ | _, Nothing -> (colspan + 2, LeftA, TDnothing) :: les | _ -> let ph s = if phony t.table.(k).(l).elem then TDnothing else s in if l = j && next_l = next_j then (1, LeftA, TDnothing) :: (colspan, CenterA, ph (TDbar None)) :: (1, LeftA, TDnothing) :: les else if l = j then (1, LeftA, ph (TDhr CenterA)) :: (colspan, RightA, ph (TDhr RightA)) :: (1, LeftA, TDnothing) :: les else if next_l = next_j then (1, LeftA, TDnothing) :: (colspan, LeftA, ph (TDhr LeftA)) :: (1, LeftA, ph (TDhr CenterA)) :: les else (colspan + 2, LeftA, ph (TDhr CenterA)) :: les in loop1 les next_l in loop1 les j in loop [] 0 in Array.of_list (List.rev les) in let hts = let rec loop hts i = if i = Array.length t.table then hts else if i = Array.length t.table - 1 && all_empty i then hts else let hts = line_elem_txt i :: hts in let hts = if i < Array.length t.table - 1 then let hts = vbars_txt (i + 1) i :: hts in let hts = if exist_several_branches i i then alone_bar_txt i :: hbars_txt i i :: hts else hts in if exist_several_branches i (i + 1) && (i < Array.length t.table - 2 || not (all_empty (i + 1))) then vbars_txt (i + 1) (i + 1) :: hbars_txt i (i + 1) :: hts else hts else hts in loop hts (i + 1) in loop [] 0 in Array.of_list (List.rev hts) (* transforming dag into table *) let ancestors d = let rec loop i = if i = Array.length d.dag then [] else let n = d.dag.(i) in if n.pare = [] then idag_of_int i :: loop (i + 1) else loop (i + 1) in loop 0 let get_children d parents = let merge_children children el = List.fold_right (fun (x, _) children -> match x with | Elem e -> let e = d.dag.(int_of_idag e) in List.fold_right (fun c children -> if List.mem c children then children else c :: children) e.chil children | Ghost _ | Nothing -> []) el children in merge_children [] parents let rec get_block t i j = if j = Array.length t.table.(i) then None else if j = Array.length t.table.(i) - 1 then let x = t.table.(i).(j) in Some ([ (x.elem, 1) ], 1) else let x = t.table.(i).(j) in let y = t.table.(i).(j + 1) in if y.span = x.span then match get_block t i (j + 1) with | Some ((x1, c1) :: list, mpc) -> let list, mpc = if x1 = x.elem then ((x1, c1 + 1) :: list, max mpc (c1 + 1)) else ((x.elem, 1) :: (x1, c1) :: list, max mpc c1) in Some (list, mpc) | _ -> assert false else Some ([ (x.elem, 1) ], 1) let group_by_common_children d list = let module O = struct type t = idag let compare = compare end in let module S = Set.Make (O) in let nlcsl = List.map (fun id -> let n = d.dag.(int_of_idag id) in let cs = List.fold_right S.add n.chil S.empty in ([ id ], cs)) list in let nlcsl = let rec loop = function | [] -> [] | (nl, cs) :: rest -> let rec loop1 beg = function | (nl1, cs1) :: rest1 -> if S.is_empty (S.inter cs cs1) then loop1 ((nl1, cs1) :: beg) rest1 else loop ((nl @ nl1, S.union cs cs1) :: List.rev_append beg rest1) | [] -> (nl, cs) :: loop rest in loop1 [] rest in loop nlcsl in List.fold_right (fun (nl, _) a -> let span = new_span_id () in List.fold_right (fun n a -> { elem = Elem n; span } :: a) nl a) nlcsl [] let copy_data d = { elem = d.elem; span = d.span } let insert_columns t nb j = let t1 = Array.make (Array.length t.table) [||] in for i = 0 to Array.length t.table - 1 do let line = t.table.(i) in let line1 = Array.make (Array.length line + nb) line.(0) in t1.(i) <- line1; let rec loop k = if k = Array.length line then () else ( if k < j then line1.(k) <- copy_data line.(k) else if k = j then for r = 0 to nb do line1.(k + r) <- copy_data line.(k) done else line1.(k + nb) <- copy_data line.(k); loop (k + 1)) in loop 0 done; { table = t1 } let rec gcd a b = if a < b then gcd b a else if b = 0 then a else gcd b (a mod b) let treat_new_row d t = let i = Array.length t.table - 1 in let rec loop t i j = match get_block t i j with | Some (parents, max_parent_colspan) -> let children = get_children d parents in let children = if children = [] then [ { elem = Nothing; span = new_span_id () } ] else List.map (fun n -> { elem = Elem n; span = new_span_id () }) children in let simple_parents_colspan = List.fold_left (fun x (_, c) -> x + c) 0 parents in if simple_parents_colspan mod List.length children = 0 then let j = j + simple_parents_colspan in let children = let cnt = simple_parents_colspan / List.length children in List.fold_right (fun d list -> let rec loop cnt list = if cnt = 1 then d :: list else copy_data d :: loop (cnt - 1) list in loop cnt list) children [] in let t, children_rest = loop t i j in (t, children @ children_rest) else let parent_colspan = List.fold_left (fun scm (_, c) -> let g = gcd scm c in scm / g * c) max_parent_colspan parents in let t, parents, _ = List.fold_left (fun (t, parents, j) (x, c) -> let to_add = (parent_colspan / c) - 1 in let t = let rec loop cc t j = if cc = 0 then t else let t = insert_columns t to_add j in loop (cc - 1) t (j + to_add + 1) in loop c t j in (t, (x, parent_colspan) :: parents, j + parent_colspan)) (t, [], j) parents in let parents = List.rev parents in let parents_colspan = parent_colspan * List.length parents in let children_colspan = List.length children in let g = gcd parents_colspan children_colspan in let t, j = let cnt = children_colspan / g in List.fold_left (fun (t, j) (_, c) -> let rec loop cc t j = if cc = 0 then (t, j) else let t = insert_columns t (cnt - 1) j in let j = j + cnt in loop (cc - 1) t j in loop c t j) (t, j) parents in let children = let cnt = parents_colspan / g in List.fold_right (fun d list -> let rec loop cnt list = if cnt = 0 then list else d :: loop (cnt - 1) list in loop cnt list) children [] in let t, children_rest = loop t i j in (t, children @ children_rest) | None -> (t, []) in loop t i 0 let down_it t i k = t.table.(Array.length t.table - 1).(k) <- t.table.(i).(k); for r = i to Array.length t.table - 2 do t.table.(r).(k) <- { elem = Ghost (new_ghost_id ()); span = new_span_id () } done (* equilibrate: in the last line, for all elem A, make fall all As, which are located at its right side above, to its line, A | i.e. transform all . into | A....... A......A *) let equilibrate t = let ilast = Array.length t.table - 1 in let last = t.table.(ilast) in let len = Array.length last in let rec loop j = if j = len then () else match last.(j).elem with | Elem x -> let rec loop1 i = if i = ilast then loop (j + 1) else let rec loop2 k = if k = len then loop1 (i + 1) else match t.table.(i).(k).elem with | Elem y when x = y -> down_it t i k; loop 0 | _ -> loop2 (k + 1) in loop2 0 in loop1 0 | Ghost _ | Nothing -> loop (j + 1) in loop 0 (* group_elem: transform all x y into x x A A A A *) let group_elem t = for i = 0 to Array.length t.table - 2 do for j = 1 to Array.length t.table.(0) - 1 do match (t.table.(i + 1).(j - 1).elem, t.table.(i + 1).(j).elem) with | Elem x, Elem y when x = y -> t.table.(i).(j).span <- t.table.(i).(j - 1).span | _ -> () done done (* group_ghost: x x x x |a |a |a |a transform all |a |b into |a |a and all x y into x x y z y y A A A A *) let group_ghost t = for i = 0 to Array.length t.table - 2 do for j = 1 to Array.length t.table.(0) - 1 do (match (t.table.(i + 1).(j - 1).elem, t.table.(i + 1).(j).elem) with | Ghost x, Ghost _ -> if t.table.(i).(j - 1).span = t.table.(i).(j).span then t.table.(i + 1).(j) <- { elem = Ghost x; span = t.table.(i + 1).(j - 1).span } | _ -> ()); match (t.table.(i).(j - 1).elem, t.table.(i).(j).elem) with | Ghost x, Ghost _ -> if t.table.(i + 1).(j - 1).elem = t.table.(i + 1).(j).elem then ( t.table.(i).(j) <- { elem = Ghost x; span = t.table.(i).(j - 1).span }; if i > 0 then t.table.(i - 1).(j).span <- t.table.(i - 1).(j - 1).span) | _ -> () done done (* group_children: transform all A A into A A x y x x *) let group_children t = for i = 0 to Array.length t.table - 1 do let line = t.table.(i) in let len = Array.length line in for j = 1 to len - 1 do if line.(j).elem = line.(j - 1).elem && line.(j).elem <> Nothing then line.(j).span <- line.(j - 1).span done done (* group_span_by_common_children: in the last line, transform all A B into A B x y x x if A and B have common children *) let group_span_by_common_children d t = let module O = struct type t = idag let compare = compare end in let module S = Set.Make (O) in let i = Array.length t.table - 1 in let line = t.table.(i) in let rec loop j cs = if j = Array.length line then () else match line.(j).elem with | Elem id -> let n = d.dag.(int_of_idag id) in let curr_cs = List.fold_right S.add n.chil S.empty in if S.is_empty (S.inter cs curr_cs) then loop (j + 1) curr_cs else ( line.(j).span <- line.(j - 1).span; loop (j + 1) (S.union cs curr_cs)) | Ghost _ | Nothing -> loop (j + 1) S.empty in loop 0 S.empty let find_same_parents t i j1 j2 j3 j4 = let rec loop i j1 j2 j3 j4 = if i = 0 then (i, j1, j2, j3, j4) else let x1 = t.(i - 1).(j1) in let x2 = t.(i - 1).(j2) in let x3 = t.(i - 1).(j3) in let x4 = t.(i - 1).(j4) in if x1.span = x4.span then (i, j1, j2, j3, j4) else let j1 = let rec loop j = if j < 0 then 0 else if t.(i - 1).(j).span = x1.span then loop (j - 1) else j + 1 in loop (j1 - 1) in let j2 = let rec loop j = if j >= Array.length t.(i) then j - 1 else if t.(i - 1).(j).span = x2.span then loop (j + 1) else j - 1 in loop (j2 + 1) in let j3 = let rec loop j = if j < 0 then 0 else if t.(i - 1).(j).span = x3.span then loop (j - 1) else j + 1 in loop (j3 - 1) in let j4 = let rec loop j = if j >= Array.length t.(i) then j - 1 else if t.(i - 1).(j).span = x4.span then loop (j + 1) else j - 1 in loop (j4 + 1) in loop (i - 1) j1 j2 j3 j4 in loop i j1 j2 j3 j4 let find_linked_children t i j1 j2 j3 j4 = let rec loop i j1 j2 j3 j4 = if i = Array.length t - 1 then (j1, j2, j3, j4) else let x1 = t.(i).(j1) in let x2 = t.(i).(j2) in let x3 = t.(i).(j3) in let x4 = t.(i).(j4) in let j1 = let rec loop j = if j < 0 then 0 else if t.(i).(j).span = x1.span then loop (j - 1) else j + 1 in loop (j1 - 1) in let j2 = let rec loop j = if j >= Array.length t.(i) then j - 1 else if t.(i).(j).span = x2.span then loop (j + 1) else j - 1 in loop (j2 + 1) in let j3 = let rec loop j = if j < 0 then 0 else if t.(i).(j).span = x3.span then loop (j - 1) else j + 1 in loop (j3 - 1) in let j4 = let rec loop j = if j >= Array.length t.(i) then j - 1 else if t.(i).(j).span = x4.span then loop (j + 1) else j - 1 in loop (j4 + 1) in loop (i + 1) j1 j2 j3 j4 in loop i j1 j2 j3 j4 let mirror_block t i1 i2 j1 j2 = for i = i1 to i2 do let line = t.(i) in let rec loop j1 j2 = if j1 >= j2 then () else let v = line.(j1) in line.(j1) <- line.(j2); line.(j2) <- v; loop (j1 + 1) (j2 - 1) in loop j1 j2 done let exch_blocks t i1 i2 j1 j2 j3 j4 = for i = i1 to i2 do let line = t.(i) in let saved = Array.copy line in for j = j1 to j2 do line.(j4 - j2 + j) <- saved.(j) done; for j = j3 to j4 do line.(j1 - j3 + j) <- saved.(j) done done let find_block_with_parents t i jj1 jj2 jj3 jj4 = let rec loop ii jj1 jj2 jj3 jj4 = let nii, njj1, njj2, njj3, njj4 = find_same_parents t i jj1 jj2 jj3 jj4 in if nii <> ii || njj1 <> jj1 || njj2 <> jj2 || njj3 <> jj3 || njj4 <> jj4 then let nii = min ii nii in let jj1, jj2, jj3, jj4 = find_linked_children t nii njj1 njj2 njj3 njj4 in if njj1 <> jj1 || njj2 <> jj2 || njj3 <> jj3 || njj4 <> jj4 then loop nii jj1 jj2 jj3 jj4 else (nii, jj1, jj2, jj3, jj4) else (ii, jj1, jj2, jj3, jj4) in loop i jj1 jj2 jj3 jj4 let push_to_right t i j1 j2 = let line = t.(i) in let rec loop j = if j = j2 then j - 1 else let ini_jj1 = match line.(j - 1).elem with | Nothing -> j - 1 | x -> let rec same_value j = if j < 0 then 0 else if line.(j).elem = x then same_value (j - 1) else j + 1 in same_value (j - 2) in let jj1 = ini_jj1 in let jj2 = j - 1 in let jj3 = j in let jj4 = match line.(j).elem with | Nothing -> j | x -> let rec same_value j = if j >= Array.length line then j - 1 else if line.(j).elem = x then same_value (j + 1) else j - 1 in same_value (j + 1) in let ii, jj1, jj2, jj3, jj4 = find_block_with_parents t i jj1 jj2 jj3 jj4 in if jj4 < j2 && jj2 < jj3 then ( exch_blocks t ii i jj1 jj2 jj3 jj4; loop (jj4 + 1)) else if jj4 < j2 && jj1 = ini_jj1 && jj2 <= jj4 then ( mirror_block t ii i jj1 jj4; loop (jj4 + 1)) else j - 1 in loop (j1 + 1) let push_to_left t i j1 j2 = let line = t.(i) in let rec loop j = if j = j1 then j + 1 else let jj1 = match line.(j).elem with | Nothing -> j | x -> let rec same_value j = if j < 0 then 0 else if line.(j).elem = x then same_value (j - 1) else j + 1 in same_value (j - 1) in let jj2 = j in let jj3 = j + 1 in let ini_jj4 = match line.(j + 1).elem with | Nothing -> j + 1 | x -> let rec same_value j = if j >= Array.length line then j - 1 else if line.(j).elem = x then same_value (j + 1) else j - 1 in same_value (j + 2) in let jj4 = ini_jj4 in let ii, jj1, jj2, jj3, jj4 = find_block_with_parents t i jj1 jj2 jj3 jj4 in if jj1 > j1 && jj2 < jj3 then ( exch_blocks t ii i jj1 jj2 jj3 jj4; loop (jj1 - 1)) else if jj1 > j1 && jj4 = ini_jj4 && jj3 >= jj1 then ( mirror_block t ii i jj1 jj4; loop (jj1 - 1)) else j + 1 in loop (j2 - 1) let fill_gap t i j1 j2 = let t1 = let t1 = Array.copy t.table in for i = 0 to Array.length t.table - 1 do t1.(i) <- Array.copy t.table.(i); for j = 0 to Array.length t1.(i) - 1 do t1.(i).(j) <- copy_data t.table.(i).(j) done done; t1 in let j2 = push_to_left t1 i j1 j2 in let j1 = push_to_right t1 i j1 j2 in if j1 = j2 - 1 then ( let line = t1.(i - 1) in let x = line.(j1).span in let y = line.(j2).span in let rec loop y j = if j >= Array.length line then () else if line.(j).span = y || t1.(i).(j).elem = t1.(i).(j - 1).elem then ( let y = line.(j).span in line.(j).span <- x; if i > 0 then t1.(i - 1).(j).span <- t1.(i - 1).(j - 1).span; loop y (j + 1)) else loop line.(j).span (j + 1) in loop y j2; Some ({ table = t1 }, true)) else None let treat_gaps t = let i = Array.length t.table - 1 in let rec loop t j = let line = t.table.(i) in if j = Array.length line then t else match line.(j).elem with | Elem _ as y -> if y = line.(j - 1).elem then loop t (j + 1) else let rec loop1 t j1 = if j1 < 0 then loop t (j + 1) else if y = line.(j1).elem then match fill_gap t i j1 j with | Some (t, ok) -> if ok then loop t 2 else loop t (j + 1) | None -> loop t (j + 1) else loop1 t (j1 - 1) in loop1 t (j - 2) | _ -> loop t (j + 1) in if Array.length t.table.(i) = 1 then t else loop t 2 let group_span_last_row t = let row = t.table.(Array.length t.table - 1) in let rec loop i = if i >= Array.length row then () else ( (match row.(i).elem with | (Elem _ | Ghost _) as x -> if x = row.(i - 1).elem then row.(i).span <- row.(i - 1).span | _ -> ()); loop (i + 1)) in loop 1 let has_phony_children phony d t = let line = t.table.(Array.length t.table - 1) in let rec loop j = if j = Array.length line then false else match line.(j).elem with | Elem x -> if phony d.dag.(int_of_idag x) then true else loop (j + 1) | _ -> loop (j + 1) in loop 0 let tablify phony no_optim no_group d = let a = ancestors d in let r = group_by_common_children d a in let t = { table = [| Array.of_list r |] } in let rec loop t = let t, new_row = treat_new_row d t in if List.for_all (fun x -> x.elem = Nothing) new_row then t else let t = { table = Array.append t.table [| Array.of_list new_row |] } in let t = if no_group && not (has_phony_children phony d t) then t else let _ = if no_optim then () else equilibrate t in let _ = group_elem t in let _ = group_ghost t in let _ = group_children t in let _ = group_span_by_common_children d t in let t = if no_optim then t else treat_gaps t in let _ = group_span_last_row t in t in loop t in loop t let fall t = for i = 1 to Array.length t.table - 1 do let line = t.table.(i) in let rec loop j = if j = Array.length line then () else match line.(j).elem with | Ghost x -> let j2 = let rec loop j = if j = Array.length line then j - 1 else match line.(j).elem with | Ghost y when y = x -> loop (j + 1) | _ -> j - 1 in loop (j + 1) in let i1 = let rec loop i = if i < 0 then i + 1 else let line = t.table.(i) in if (j = 0 || line.(j - 1).span <> line.(j).span) && (j2 = Array.length line - 1 || line.(j2 + 1).span <> line.(j2).span) then loop (i - 1) else i + 1 in loop (i - 1) in let i1 = if i1 = i then i1 else if i1 = 0 then i1 else if t.table.(i1).(j).elem = Nothing then i1 else i in if i1 < i then ( for k = i downto i1 + 1 do for j = j to j2 do t.table.(k).(j).elem <- t.table.(k - 1).(j).elem; if k < i then t.table.(k).(j).span <- t.table.(k - 1).(j).span done done; for l = j to j2 do if i1 = 0 || t.table.(i1 - 1).(l).elem = Nothing then t.table.(i1).(l).elem <- Nothing else t.table.(i1).(l) <- (if l = j || t.table.(i1 - 1).(l - 1).span <> t.table.(i1 - 1).(l).span then { elem = Ghost (new_ghost_id ()); span = new_span_id () } else copy_data t.table.(i1).(l - 1)) done); loop (j2 + 1) | _ -> loop (j + 1) in loop 0 done let fall2_cool_right t i1 i2 j1 j2 = let span = t.table.(i2 - 1).(j1).span in for i = i2 - 1 downto 0 do for j = j1 to j2 - 1 do t.table.(i).(j) <- (if i - i2 + i1 >= 0 then t.table.(i - i2 + i1).(j) else { elem = Nothing; span = new_span_id () }) done done; for i = Array.length t.table - 1 downto 0 do for j = j2 to Array.length t.table.(i) - 1 do t.table.(i).(j) <- (if i - i2 + i1 >= 0 then t.table.(i - i2 + i1).(j) else { elem = Nothing; span = new_span_id () }) done done; let old_span = t.table.(i2 - 1).(j1).span in let rec loop j = if j = Array.length t.table.(i2 - 1) then () else if t.table.(i2 - 1).(j).span = old_span then ( t.table.(i2 - 1).(j).span <- span; loop (j + 1)) in loop j1 let fall2_cool_left t i1 i2 j1 j2 = let span = t.table.(i2 - 1).(j2).span in for i = i2 - 1 downto 0 do for j = j1 + 1 to j2 do t.table.(i).(j) <- (if i - i2 + i1 >= 0 then t.table.(i - i2 + i1).(j) else { elem = Nothing; span = new_span_id () }) done done; for i = Array.length t.table - 1 downto 0 do for j = j1 downto 0 do t.table.(i).(j) <- (if i - i2 + i1 >= 0 then t.table.(i - i2 + i1).(j) else { elem = Nothing; span = new_span_id () }) done done; let old_span = t.table.(i2 - 1).(j2).span in let rec loop j = if j < 0 then () else if t.table.(i2 - 1).(j).span = old_span then ( t.table.(i2 - 1).(j).span <- span; loop (j - 1)) in loop j2 let do_fall2_right t i1 i2 j1 j2 = let i3 = let rec loop_i i = if i < 0 then 0 else let rec loop_j j = if j = Array.length t.table.(i) then loop_i (i - 1) else match t.table.(i).(j).elem with | Nothing -> loop_j (j + 1) | _ -> i + 1 in loop_j j2 in loop_i (Array.length t.table - 1) in let new_height = i3 + i2 - i1 in let t = if new_height > Array.length t.table then let rec loop cnt t = if cnt = 0 then t else let new_line = Array.init (Array.length t.table.(0)) (fun _ -> { elem = Nothing; span = new_span_id () }) in let t = { table = Array.append t.table [| new_line |] } in loop (cnt - 1) t in loop (new_height - Array.length t.table) t else t in fall2_cool_right t i1 i2 j1 j2; t let do_fall2_left t i1 i2 j1 j2 = let i3 = let rec loop_i i = if i < 0 then 0 else let rec loop_j j = if j < 0 then loop_i (i - 1) else match t.table.(i).(j).elem with | Nothing -> loop_j (j - 1) | _ -> i + 1 in loop_j j1 in loop_i (Array.length t.table - 1) in let new_height = i3 + i2 - i1 in let t = if new_height > Array.length t.table then let rec loop cnt t = if cnt = 0 then t else let new_line = Array.init (Array.length t.table.(0)) (fun _ -> { elem = Nothing; span = new_span_id () }) in let t = { table = Array.append t.table [| new_line |] } in loop (cnt - 1) t in loop (new_height - Array.length t.table) t else t in fall2_cool_left t i1 i2 j1 j2; t let do_shorten_too_long t i1 j1 j2 = for i = i1 to Array.length t.table - 2 do for j = j1 to j2 - 1 do t.table.(i).(j) <- t.table.(i + 1).(j) done done; let i = Array.length t.table - 1 in for j = j1 to j2 - 1 do t.table.(i).(j) <- { elem = Nothing; span = new_span_id () } done; t let try_fall2_right t i j = match t.table.(i).(j).elem with | Ghost _ -> let i1 = let rec loop i = if i < 0 then 0 else match t.table.(i).(j).elem with | Ghost _ -> loop (i - 1) | _ -> i + 1 in loop (i - 1) in let separated1 = let rec loop i = if i < 0 then true else if j > 0 && t.table.(i).(j - 1).span = t.table.(i).(j).span then false else loop (i - 1) in loop (i1 - 1) in let j2 = let x = t.table.(i).(j).span in let rec loop j2 = if j2 = Array.length t.table.(i) then j2 else match t.table.(i).(j2) with | { elem = Ghost _; span = y } when y = x -> loop (j2 + 1) | _ -> j2 in loop (j + 1) in let separated2 = let rec loop i = if i = Array.length t.table then true else if j2 = Array.length t.table.(i) then false else if t.table.(i).(j2 - 1).span = t.table.(i).(j2).span then false else loop (i + 1) in loop (i + 1) in if (not separated1) || not separated2 then None else Some (do_fall2_right t i1 (i + 1) j j2) | _ -> None let try_fall2_left t i j = match t.table.(i).(j).elem with | Ghost _ -> let i1 = let rec loop i = if i < 0 then 0 else match t.table.(i).(j).elem with | Ghost _ -> loop (i - 1) | _ -> i + 1 in loop (i - 1) in let separated1 = let rec loop i = if i < 0 then true else if j < Array.length t.table.(i) - 1 && t.table.(i).(j).span = t.table.(i).(j + 1).span then false else loop (i - 1) in loop (i1 - 1) in let j1 = let x = t.table.(i).(j).span in let rec loop j1 = if j1 < 0 then j1 else match t.table.(i).(j1) with | { elem = Ghost _; span = y } when y = x -> loop (j1 - 1) | _ -> j1 in loop (j - 1) in let separated2 = let rec loop i = if i = Array.length t.table then true else if j1 < 0 then false else if t.table.(i).(j1).span = t.table.(i).(j1 + 1).span then false else loop (i + 1) in loop (i + 1) in if (not separated1) || not separated2 then None else Some (do_fall2_left t i1 (i + 1) j1 j) | _ -> None let try_shorten_too_long t i j = match t.table.(i).(j).elem with | Ghost _ -> let j2 = let x = t.table.(i).(j).span in let rec loop j2 = if j2 = Array.length t.table.(i) then j2 else match t.table.(i).(j2) with | { elem = Ghost _; span = y } when y = x -> loop (j2 + 1) | _ -> j2 in loop (j + 1) in let i1 = let rec loop i = if i = Array.length t.table then i else match t.table.(i).(j).elem with Elem _ -> loop (i + 1) | _ -> i in loop (i + 1) in let i2 = let rec loop i = if i = Array.length t.table then i else match t.table.(i).(j).elem with Nothing -> loop (i + 1) | _ -> i in loop i1 in let separated_left = let rec loop i = if i = i2 then true else if j > 0 && t.table.(i).(j).span = t.table.(i).(j - 1).span then false else loop (i + 1) in loop i in let separated_right = let rec loop i = if i = i2 then true else if j2 < Array.length t.table.(i) && t.table.(i).(j2 - 1).span = t.table.(i).(j2).span then false else loop (i + 1) in loop i in if (not separated_left) || not separated_right then None else if i2 < Array.length t.table then None else Some (do_shorten_too_long t i j j2) | _ -> None let fall2_right t = let rec loop_i i t = if i <= 0 then t else let rec loop_j j t = if j < 0 then loop_i (i - 1) t else match try_fall2_right t i j with | Some t -> loop_i (Array.length t.table - 1) t | None -> loop_j (j - 1) t in loop_j (Array.length t.table.(i) - 2) t in loop_i (Array.length t.table - 1) t let fall2_left t = let rec loop_i i t = if i <= 0 then t else let rec loop_j j t = if j >= Array.length t.table.(i) then loop_i (i - 1) t else match try_fall2_left t i j with | Some t -> loop_i (Array.length t.table - 1) t | None -> loop_j (j + 1) t in loop_j 1 t in loop_i (Array.length t.table - 1) t let shorten_too_long t = let rec loop_i i t = if i <= 0 then t else let rec loop_j j t = if j >= Array.length t.table.(i) then loop_i (i - 1) t else match try_shorten_too_long t i j with | Some t -> loop_i (Array.length t.table - 1) t | None -> loop_j (j + 1) t in loop_j 1 t in loop_i (Array.length t.table - 1) t (* top_adjust: deletes all empty rows that might have appeared on top of the table after the falls *) let top_adjust t = let di = let rec loop i = if i = Array.length t.table then i else let rec loop_j j = if j = Array.length t.table.(i) then loop (i + 1) else if t.table.(i).(j).elem <> Nothing then i else loop_j (j + 1) in loop_j 0 in loop 0 in if di > 0 then ( for i = 0 to Array.length t.table - 1 - di do t.table.(i) <- t.table.(i + di) done; { table = Array.sub t.table 0 (Array.length t.table - di) }) else t (* bottom_adjust: deletes all empty rows that might have appeared on bottom of the table after the falls *) let bottom_adjust t = let last_i = let rec loop i = if i < 0 then i else let rec loop_j j = if j = Array.length t.table.(i) then loop (i - 1) else if t.table.(i).(j).elem <> Nothing then i else loop_j (j + 1) in loop_j 0 in loop (Array.length t.table - 1) in if last_i < Array.length t.table - 1 then { table = Array.sub t.table 0 (last_i + 1) } else t (* invert *) let invert_dag d = let d = { dag = Array.copy d.dag } in for i = 0 to Array.length d.dag - 1 do let n = d.dag.(i) in d.dag.(i) <- { pare = List.map (fun x -> x) n.chil; valu = n.valu; chil = List.map (fun x -> x) n.pare; } done; d let invert_table t = let t' = { table = Array.copy t.table } in let len = Array.length t.table in for i = 0 to len - 1 do t'.table.(i) <- Array.init (Array.length t.table.(0)) (fun j -> let d = t.table.(len - 1 - i).(j) in { elem = d.elem; span = d.span }); if i < len - 1 then for j = 0 to Array.length t'.table.(i) - 1 do t'.table.(i).(j).span <- t.table.(len - 2 - i).(j).span done done; t' (* main *) let table_of_dag phony no_optim invert no_group d = let d = if invert then invert_dag d else d in let t = tablify phony no_optim no_group d in let t = if invert then invert_table t else t in let _ = fall t in fall2_right t |> fall2_left |> shorten_too_long |> top_adjust |> bottom_adjust