diff --git a/.merlin b/.merlin deleted file mode 100644 index 85d3a654aab398a822e7c90a0fc7cfca3ae8a9cd..0000000000000000000000000000000000000000 --- a/.merlin +++ /dev/null @@ -1,10 +0,0 @@ -PKG lutils rdbg lustre-v6 lutin - -S src/ -S lib/sasacore/ -S lib/sasa/ -S lib/algo/ - -B _build/default/lib/algo/.algo.objs/* -B _build/default/lib/sasa/.sasa.objs/* -B _build/default/lib/sasacore/.sasacore.objs/* diff --git a/lib/algo/algo.ml b/lib/algo/algo.ml index 91d0bed03bf7664114694c28beea997162baa880..de8f960747589e2447d429589b252089ddddbdc3 100644 --- a/lib/algo/algo.ml +++ b/lib/algo/algo.ml @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 13/10/2020 (at 15:55) by Erwan Jahier> *) +(* Time-stamp: <modified the 09/04/2021 (at 10:04) by Erwan Jahier> *) open Sasacore (* Process programmer API *) @@ -144,11 +144,15 @@ let max_degree = Register.max_degree let is_cyclic = Register.is_cyclic let is_connected = Register.is_connected let is_directed = Register.is_directed -let is_tree = Register.is_tree -let height = Register.height let links_number = Register.links_number let diameter = Register.diameter +let is_tree = Register.is_tree +let is_in_tree = Register.is_in_tree +let is_out_tree = Register.is_out_tree +let height = Register.height +let sub_tree_size = Register.sub_tree_size +let parent = Register.parent (* let pid n = n.pid diff --git a/lib/algo/algo.mli b/lib/algo/algo.mli index b2195f0cb6587155c27488b580a6b9bbcc84c3c7..4c48f6d23f7435f0c29b476d6887bdb8d3e2d353 100644 --- a/lib/algo/algo.mli +++ b/lib/algo/algo.mli @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 04/11/2020 (at 08:15) by Erwan Jahier> *) +(* Time-stamp: <modified the 09/04/2021 (at 10:39) by Erwan Jahier> *) (** {1 The Algorithm programming Interface} A SASA process is an instance of an algorithm defined via this @@ -22,15 +22,16 @@ type 's enable_fun = 's -> 's neighbor list -> action list type 's step_fun = 's -> 's neighbor list -> action -> 's (** [enable_fun] and [step_fun] have the same arguments in input: - - The first argument holds the current state of the process. As it is - polymorphic (['s]), algorithms designers can put anything they need - into this state (an integer, a structure, etc.). The only constraint - is that all algorithms should use the same type. + - The first argument holds the current state of the process. As it + is polymorphic (['s]), algorithms designers can put anything they + need into this state (an integer, a structure, etc.). The only + constraint is that all algorithms should use the same type. - - The second argument holds the process neighbors. Note that SASA - processes, that live in graph nodes, can only access to their - immediate neighbors. From each neighbor, a process can access to - various information (cf [state], [reply], and [weight] functions below). + - The second argument holds the process neighbors (its successors + in the graph). Note that SASA processes, that live in graph nodes, + can only access to their immediate neighbors. From each neighbor, + a process can access to various information (cf [state], [reply], + and [weight] functions below). [enable_fun] returns the list of enable actions. @@ -82,7 +83,7 @@ val state : 's neighbor -> 's (** Returns the neighbor channel number, that let this neighbor access to the content of the current process, if its neighbor can access - it. The channel number is the rank,starting at 0, in the + it. The channel number is the rank, starting at 0, in the neighbors' list. Returns -1 if the neighbor can not access to the process, which may happen in directed graphs. @@ -111,11 +112,26 @@ val max_degree : unit -> int val is_directed : unit -> bool val is_cyclic : unit -> bool val is_connected : unit -> bool -val is_tree : unit -> bool -val height : unit -> (string (* the node id *) -> int) option val links_number : unit -> int val diameter : unit -> int +(** {3 Trees} *) +val is_tree : unit -> bool +val is_in_tree : unit -> bool +val is_out_tree : unit -> bool + +(** The 3 functions below work for in_tree or out_tree only. *) + +(* maps each node to the size of the corresponding sub-tree *) +val sub_tree_size : string (* the node id *) -> int + +(** maps each node to its height in the tree *) +val height : string (* the node id *) -> int + +(** maps each node to the channel number of its parent, and to None + for the tree root. *) +val parent : string (* the node id *) -> int option + (** It is possible to set some global parameters in the dot file using graph attributes. This function allows one the get their values. *) diff --git a/lib/sasacore/main.ml b/lib/sasacore/main.ml index 5632ed6b0fe199720dbd3385e628f65231a0ccdd..02dbcda708a4206f45ba5281ec1c2c454bac0d49 100644 --- a/lib/sasacore/main.ml +++ b/lib/sasacore/main.ml @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 02/04/2021 (at 15:47) by Erwan Jahier> *) +(* Time-stamp: <modified the 07/04/2021 (at 11:50) by Erwan Jahier> *) open Register @@ -10,17 +10,6 @@ let (update_env_with_init : 'v Env.t -> 'v Process.t list -> 'v Env.t) = in List.fold_left aux e pl -(** Returns the channel number that let [p_neighbor] access to the - content of [p], if [p] is a neighbor of [p_neighbor]. Returns -1 if - [p] is not a neigbhbor of [p_neigbor], which can happen in directed - graphs. *) -let (reply: Topology.t -> string -> string -> int) = - fun g p p_neighbor -> - let rec f i = function - | [] -> (-1) (* may happen in directed graphs *) - | (_w,x)::t -> if x=p then i else f (i+1) t - in - f 0 (g.succ p_neighbor) let (get_neighors: Topology.t -> Topology.node_id -> 'v -> 'v Register.neighbor list) = fun g source_id init -> @@ -32,7 +21,7 @@ let (get_neighors: Topology.t -> Topology.node_id -> 'v -> 'v Register.neighbor state = init; pid = node.id; spid = source_id; - reply = (fun () -> reply g source_id neighbor_id); + reply = (fun () -> Topology.reply g source_id neighbor_id); weight = (fun () -> w) } ) @@ -208,14 +197,7 @@ let (make : bool -> string array -> 'v t) = let nidl = List.map (fun n -> n.Topology.id) nl in let nstr = String.concat "," nidl in - Register.set_card (fun () -> List.length nl); - Register.set_degrees (fun () -> Topology.get_degree g); - Register.set_mean_deg (fun () -> Topology.get_mean_degree g); - Register.set_is_connected_cyclic (fun () -> Topology.is_connected_and_cyclic g); - Register.set_height (Topology.get_height g); - Register.set_links_number (fun () -> Topology.get_nb_link g); - Register.set_diameter (fun () -> Diameter.get g); - Register.set_is_directed (fun () -> g.directed); + Register.set_topology g; List.iter (fun (n,v) -> Register.set_graph_attribute n v) g.attributes; Register.verbose_level := args.verbose; diff --git a/lib/sasacore/register.ml b/lib/sasacore/register.ml index e92a5b3243642c6eef90d736f2d1a5315a5fe518..5ba3995665eafe46cace5ca0f33d07f597282a9f 100644 --- a/lib/sasacore/register.ml +++ b/lib/sasacore/register.ml @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 07/04/2021 (at 10:16) by Erwan Jahier> *) +(* Time-stamp: <modified the 08/04/2021 (at 22:03) by Erwan Jahier> *) type 's neighbor = { state: 's ; @@ -38,6 +38,8 @@ type 's internal_tables = { mutable is_cyclic : bool option; mutable is_connected : bool option; mutable is_tree : bool option; + mutable is_in_tree : bool option; + mutable is_out_tree : bool option; mutable is_directed : bool option; mutable height : (string -> int) option; mutable sub_tree_size: (string -> int) option; @@ -68,6 +70,8 @@ let (tbls:'s internal_tables) = { is_cyclic = None; is_connected = None; is_tree = None; + is_in_tree = None; + is_out_tree = None; is_directed = None; height = None; parent = None; @@ -223,8 +227,7 @@ let (max_degree : unit -> int) = fun () -> let (is_cyclic : unit -> bool) = fun () -> match tbls.is_cyclic with | None -> - let connect, cyclic = Topology.is_connected_and_cyclic (get_topology()) in - tbls.is_connected <- Some connect; + let cyclic = Topology.is_cyclic (get_topology()) in tbls.is_cyclic <- Some cyclic; cyclic | Some b -> b @@ -232,54 +235,66 @@ let (is_cyclic : unit -> bool) = fun () -> let (is_connected : unit -> bool) = fun () -> match tbls.is_connected with | None -> - let connect, cyclic = Topology.is_connected_and_cyclic (get_topology()) in + let connect = Topology.is_connected (get_topology()) in tbls.is_connected <- Some connect; - tbls.is_cyclic <- Some cyclic; connect | Some b -> b let (is_tree : unit -> bool) = fun () -> match tbls.is_tree with | None -> - let b = (not (is_cyclic ()) && (is_connected ())) in + let b = Topology.is_tree (get_topology()) in tbls.is_tree <- Some b; b | Some b -> b -exception Not_a_tree +let (is_in_tree : unit -> bool) = fun () -> + match tbls.is_in_tree with + | None -> + let b = Topology.is_in_tree (get_topology()) in + tbls.is_in_tree <- Some b; + b + | Some b -> b -let height : (unit -> string -> int) = - fun () -> +let (is_out_tree : unit -> bool) = fun () -> + match tbls.is_out_tree with + | None -> + let b = Topology.is_out_tree (get_topology()) in + tbls.is_out_tree <- Some b; + b + | Some b -> b + +let not_a_tree () = failwith "The graph is not a tree" + +let height : (string -> int) = + fun pid -> if is_tree () then ( match tbls.height with - | Some h -> h + | Some h -> h pid | None -> let h = Topology.get_height (get_topology ()) in - tbls.height <- Some h; h + tbls.height <- Some h; h pid ) - else raise Not_a_tree + else not_a_tree () -let sub_tree_size : (unit -> string -> int) = - fun () -> +let sub_tree_size : (string -> int) = + fun pid -> if is_tree () then ( match tbls.sub_tree_size with - | Some s -> s + | Some s -> s pid | None -> let s = Topology.get_sub_tree_size (get_topology ()) in - tbls.sub_tree_size <- Some s; s + tbls.sub_tree_size <- Some s; s pid ) - else raise Not_a_tree + else not_a_tree () -let parent : (unit -> string -> int option) = - fun () -> - if is_tree () then ( +let parent : (string -> int option) = + fun pid -> match tbls.parent with - | Some p -> p + | Some p -> p pid | None -> let p = Topology.get_parent (get_topology ()) in - tbls.parent <- Some p; p - ) - else raise Not_a_tree + tbls.parent <- Some p; p pid let (links_number : unit -> int) = diff --git a/lib/sasacore/register.mli b/lib/sasacore/register.mli index e405b94d6cbb8e3020a9760482a03e6ca7446e7f..b33fed0e2bc7eeba40fd309e2d8080b897f4c6e7 100644 --- a/lib/sasacore/register.mli +++ b/lib/sasacore/register.mli @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 13/10/2020 (at 15:36) by Erwan Jahier> *) +(* Time-stamp: <modified the 08/04/2021 (at 21:55) by Erwan Jahier> *) (** This module duplicates and extends the Algo module with get_* functions. @@ -43,15 +43,16 @@ val mean_degree : unit -> float val max_degree : unit -> int val is_cyclic : unit -> bool val is_connected : unit -> bool -val is_tree : unit -> bool -val is_directed : unit -> bool - -(** If height () = None, then the graph doesn't have a height (because it isn't a tree) - Otherwise, height () = Some h.*) -val height : unit -> (string -> int) option val links_number : unit -> int val diameter : unit -> int +val is_tree : unit -> bool +val is_in_tree : unit -> bool +val is_out_tree : unit -> bool +val sub_tree_size : string -> int +val parent : string -> int option +val is_directed : unit -> bool +val height : string -> int val verbose_level: int ref val get_graph_attribute : string -> string @@ -74,15 +75,8 @@ val get_copy_value : unit -> ('s -> 's) val to_string : 's -> string (** Those are called by sasa once the graph has been parsed *) -val set_card : (unit -> int) -> unit -val set_degrees : (unit -> int*int) -> unit -val set_mean_deg : (unit -> float) -> unit -val set_is_connected_cyclic : (unit -> bool*bool) -> unit -val set_is_directed : (unit -> bool) -> unit +val set_topology : Topology.t -> unit type node_id = string (* cf topology.mli *) -val set_height : (node_id -> int) -> unit -val set_links_number : (unit -> int) -> unit -val set_diameter : (unit -> int) -> unit val set_graph_attribute : string -> string -> unit val graph_attribute_list: unit -> (string * string) list diff --git a/lib/sasacore/topology.ml b/lib/sasacore/topology.ml index d40bcbf25dd639dd30ffa014c6a5c3fc2eef1243..a59e6cb816d6935770743972db36b5d14b33032a 100644 --- a/lib/sasacore/topology.ml +++ b/lib/sasacore/topology.ml @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 09/11/2020 (at 21:58) by Erwan Jahier> *) +(* Time-stamp: <modified the 09/04/2021 (at 10:48) by Erwan Jahier> *) open Graph open Graph.Dot_ast @@ -14,6 +14,7 @@ type node = { type t = { nodes: node list; succ: node_id -> (int * node_id) list; + pred: node_id -> node_id list; of_id: node_id -> node; directed:bool; attributes: (string * string) list; @@ -22,12 +23,15 @@ type t = { type node_info_t = (string, node) Hashtbl.t let node_info:node_info_t = Hashtbl.create 100 -type node_succ_t = (string, (int * node_id) list) Hashtbl.t +type node_succ_t = (string, int * node_id) Hashtbl.t let node_succ:node_succ_t = Hashtbl.create 100 +type node_pred_t = (string, node_id) Hashtbl.t +let node_pred:node_pred_t = Hashtbl.create 100 let clean_tbl () = Hashtbl.clear node_info; - Hashtbl.clear node_succ + Hashtbl.clear node_succ; + Hashtbl.clear node_pred let (of_id:Dot_ast.id -> string) = function Ident str | Html str | Number str | String str -> str @@ -89,7 +93,7 @@ let (do_stmt: bool -> node list * attrs -> Dot_ast.stmt -> node list * attrs) = let id = of_node_id node_id in let inits = get_init dot_attrs in let node = { id=id ; file = get_file node_id dot_attrs ; init = inits } in - Hashtbl.replace node_info id node; + Hashtbl.add node_info id node; node::n, attrs | Edge_stmt (dot_node, nodes, dot_attrs) -> let node = of_node dot_node in @@ -110,12 +114,12 @@ let (do_stmt: bool -> node list * attrs -> Dot_ast.stmt -> node list * attrs) = if n1 = n2 then failwith (Printf.sprintf "Bad topology: %s can not ne a neighbor of itself!" n1); - let pn1 = try Hashtbl.find node_succ n1 with Not_found -> [] in - let pn2 = try Hashtbl.find node_succ n2 with Not_found -> [] in + let pn1 = Hashtbl.find_all node_succ n1 in + let pn2 = Hashtbl.find_all node_succ n2 in if not (List.mem (weight,n2) pn1) then - Hashtbl.replace node_succ n1 ((weight,n2)::pn1); + Hashtbl.add node_succ n1 (weight,n2); if not directed && not (List.mem (weight,n1) pn2) then - Hashtbl.replace node_succ n2 ((weight,n1)::pn2); + Hashtbl.add node_succ n2 (weight,n1); n2 in ignore (List.fold_left add_edge node nodes); @@ -133,10 +137,20 @@ let (read: string -> t) = fun f -> clean_tbl (); let dot_file = Graph.Dot.parse_dot_ast f in assert (not dot_file.strict); - let nodes, attrs = List.fold_left (do_stmt dot_file.digraph) ([], []) dot_file.stmts in + let nodes, attrs = + List.fold_left (do_stmt dot_file.digraph) ([], []) dot_file.stmts + in + Hashtbl.iter + (fun pid (_, pid_succ) -> + Hashtbl.add node_pred pid_succ pid + ) + node_succ; + let succ str = Hashtbl.find_all node_succ str in + let pred str = Hashtbl.find_all node_pred str in { nodes = List.rev nodes; - succ = (fun str -> try Hashtbl.find node_succ str with Not_found -> []); + succ = succ; + pred = pred; of_id = (fun str -> try Hashtbl.find node_info str with Not_found -> failwith (str^ " unknown node id") ); @@ -153,7 +167,8 @@ let (to_adjacency: t -> bool array array) = List.iteri (fun i n -> Hashtbl.add rank_node_tbl n.id i) t.nodes; List.iteri (fun i n -> - List.iter (fun (_,target) -> m.(i).(rank_node target) <- true) (t.succ n.id) + List.iter (fun (_,target) -> + m.(i).(rank_node target) <- true) (t.succ n.id) ) t.nodes; m @@ -163,21 +178,29 @@ let (get_degree: t -> int*int) = match t.nodes with | [] -> 0,0 | n::tl -> - let node_deg n = List.length (t.succ (n.id)) in + let node_deg n = + if t.directed then + List.length (t.succ (n.id)) + List.length (t.pred (n.id)) + else + List.length (t.succ (n.id)) + in let d0 = node_deg n in let dmin, dmax = List.fold_left - (fun (d_min,d_max) n -> - (min (node_deg n) d_min, max (node_deg n) d_max) - ) - (d0, d0) - tl + (fun (d_min,d_max) n -> + (min (node_deg n) d_min, max (node_deg n) d_max) + ) + (d0, d0) + tl in - if t.directed then 2*dmin, 2*dmax else dmin, dmax + dmin, dmax let (get_nb_link: t -> int) = fun t -> let res = List.fold_left - (fun acc n -> ((List.length (t.succ n.id))) + acc) 0 t.nodes + (fun acc n -> + let succ = String.concat "," (List.map snd (t.succ n.id)) in + Printf.printf "%s->%s\n%!" n.id succ; + ((List.length (t.succ n.id))) + acc) 0 t.nodes in if t.directed then res else res/2 @@ -185,6 +208,42 @@ let (get_mean_degree : t -> float) = fun t -> 2.0 *. (float_of_int (get_nb_link t)) /. (float_of_int (List.length t.nodes)) +type color = W | G | B +exception Cycle +let directed_is_cyclic : t -> bool = + fun g -> + assert (g.directed); + let t = Hashtbl.create (List.length g.nodes) in + let nodes = List.map (fun n -> n.id) g.nodes in + let color pid = match Hashtbl.find_opt t pid with + Some c -> c | None -> assert false + in + List.iter (fun n -> Hashtbl.add t n W) nodes; + let rec visit pid = + match color pid with + | G -> raise Cycle + | B -> () + | W -> + Hashtbl.replace t pid G; + List.iter visit (g.pred pid); + Hashtbl.replace t pid B + in + try List.iter visit nodes; false + with Cycle -> true + +let is_connected : t -> bool = + fun g -> + let visited = Hashtbl.create 0 in + let rec f acc pid = + if Hashtbl.mem visited pid then acc else + (Hashtbl.add visited pid pid; + List.fold_left f (pid::acc) + (List.rev_append (List.map snd (g.succ pid)) (g.pred pid)) + ) + in + let accessible = f [] (List.hd g.nodes).id in + List.length accessible = List.length g.nodes + let bfs : (t -> string -> bool * string list) = fun t n -> let q = Queue.create () in @@ -206,26 +265,123 @@ let bfs : (t -> string -> bool * string list) = parents ) ) !parent (t.succ node) - done; (!cyclic, !discovered) -let is_connected_and_cyclic : t -> bool*bool = - fun t -> match t.nodes with - | [] -> (false,false) +let non_directed_is_cyclic : t -> bool = + fun t -> + assert(not t.directed); + match t.nodes with + | [] -> false | hd::_ -> - let (cyclic,bfs_nodes) = (bfs t hd.id) in - ((List.compare_lengths t.nodes bfs_nodes) = 0, cyclic) - -let rec height : string list -> t -> string -> int = - fun parents t n -> - (List.fold_left (fun h (_,succ) -> - if List.mem succ parents then - h - else - max h (height (n::parents) t succ)) (-1) (t.succ n)) + 1 - -let get_height : t -> string -> int = - fun t -> - height ([]) t + let (cyclic, _bfs_nodes) = (bfs t hd.id) in + cyclic + +let is_cyclic : t -> bool = + fun g -> + if g.directed then directed_is_cyclic g else non_directed_is_cyclic g + +let is_tree : t -> bool = + fun g -> + (not (is_cyclic g)) && (is_connected g) + + +(** Returns the channel number that let [p_neighbor] access to the + content of [p], if [p] is a neighbor of [p_neighbor]. Returns -1 if + [p] is not a neigbhbor of [p_neigbor], which can happen in directed + graphs. *) +let (reply: t -> string -> string -> int) = + fun g p p_neighbor -> + let rec f i = function + | [] -> (-1) (* may happen in directed graphs *) + | (_w,x)::t -> if x=p then i else f (i+1) t + in + f 0 (g.succ p_neighbor) + +let (reply_pred: t -> string -> string -> int) = + fun g p p_neighbor -> + let rec f i = function + | [] -> (-1) (* may happen in directed graphs *) + | x::t -> if x=p then i else f (i+1) t + in + f 0 (g.pred p_neighbor) + +let is_in_tree g = + (is_tree g) && + (List.for_all (fun n -> List.length (g.pred n.id) <= 1) g.nodes) + +let is_out_tree g = + (is_tree g) && + (List.for_all (fun n -> List.length (g.succ n.id) <= 1) g.nodes) + +let not_a_io_tree () = failwith "The graph is not an in-tree nor an out-tree" + +let get_parent = fun g pid -> + let in_tree = is_in_tree g in + let out_tree = is_out_tree g in + if not in_tree && not out_tree then + not_a_io_tree () + else + let succ,reply = + if out_tree then + (fun pid -> List.map snd (g.succ pid)), reply_pred + else + (fun pid -> g.pred pid), reply + in + match succ pid with + | [] -> None + | [par] -> Some (reply g pid par) + | l -> + Printf.printf "%s->%s\n%!" pid (String.concat "," l); + assert false + +let get_sub_tree_size g pid = + let in_tree = is_in_tree g in + let out_tree = is_out_tree g in + if not in_tree && not out_tree then + not_a_io_tree () + else + let succ = + if in_tree then + (fun pid -> List.map snd (g.succ pid)) + else + (fun pid -> g.pred pid) + in + let visited = Hashtbl.create 0 in + let rec f acc pid = + if Hashtbl.mem visited pid then acc else + (Hashtbl.add visited pid pid; + List.fold_left + (fun acc pid -> f acc pid) + (acc+1) + (succ pid) + ) + in + f 0 pid + +let rec height : (string -> string list) -> string list -> string -> int = + fun succ parents n -> + (List.fold_left + (fun h pid -> + if List.mem pid parents then h else + max h (height succ (n::parents) pid) + ) + (-1) + (succ n) + ) + 1 + +let get_height : t -> string -> int = + fun g -> + let in_tree = is_in_tree g in + let out_tree = is_out_tree g in + if not in_tree && not out_tree then + not_a_io_tree () + else + let succ = + if out_tree then + (fun pid -> List.map snd (g.succ pid)) + else + (fun pid -> g.pred pid) + in + height succ [] diff --git a/lib/sasacore/topology.mli b/lib/sasacore/topology.mli index a27f0e6dde7311ded6171929042e486bf8465fb5..e87887ff9e70f53f084dfdda118e352f84cdf773 100644 --- a/lib/sasacore/topology.mli +++ b/lib/sasacore/topology.mli @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 31/08/2020 (at 16:45) by Erwan Jahier> *) +(* Time-stamp: <modified the 09/04/2021 (at 10:48) by Erwan Jahier> *) (** {1 Topology: internal representation of Graphs } *) @@ -13,6 +13,7 @@ type node = { type t = { nodes: node list; (** *) succ: node_id -> (int * node_id) list; (** get neighbors, with weight *) + pred: node_id -> node_id list; of_id: node_id -> node; (** *) directed:bool; (** true if the graph is directed *) attributes: (string * string) list (** (name, value) list of graph attributes *) @@ -22,11 +23,22 @@ type t = { val read: string -> t (** {1 Various eponymous util functions } *) - + val to_adjacency: t -> bool array array val get_nb_link: t -> int val get_mean_degree : t -> float -val is_connected_and_cyclic : t -> bool * bool -val height : string list -> t -> string -> int +val is_connected : t -> bool +val is_cyclic : t -> bool +val is_tree : t -> bool +val is_in_tree : t -> bool +val is_out_tree : t -> bool val get_height : t -> string -> int +val get_parent : t -> string -> int option +val get_sub_tree_size : t -> string -> int val get_degree: t -> int * int + +(** [reply g p p_neighbor] returns the channel number that let + [p_neighbor] access to the content of [p], if [p] is a neighbor of + [p_neighbor]. Returns -1 if [p] is not a neighbor of [p_neigbor], + which can happen in directed graphs. *) +val reply: t -> string -> string -> int diff --git a/test/Makefile.dot b/test/Makefile.dot index 03b8593a87b9acb786a7413572cfb8d37f296aa1..e67483c7cf3f3706cf3029929444627fcfacf2e5 100644 --- a/test/Makefile.dot +++ b/test/Makefile.dot @@ -1,4 +1,4 @@ -# Time-stamp: <modified the 04/12/2020 (at 11:14) by Erwan Jahier> +# Time-stamp: <modified the 09/04/2021 (at 10:48) by Erwan Jahier> # Rules to generate various dot files. @@ -33,8 +33,20 @@ ring%.dot: gg ring -n $* -o $@ gg-deco $(DECO_PATTERN) $@ -o $@ -dtree%.dot: - gg tree -dir -n $* -o $@ +tree%.dot: + gg tree -n $* -o $@ + gg-deco $(DECO_PATTERN) $@ -o $@ + +intree%.dot: + gg tree --in-tree -n $* -o $@ + gg-deco $(DECO_PATTERN) $@ -o $@ + +outtree%.dot: + gg tree --out-tree -n $* -o $@ + gg-deco $(DECO_PATTERN) $@ -o $@ + +inouttree%.dot: + gg tree --in-out-tree -n $* -o $@ gg-deco $(DECO_PATTERN) $@ -o $@ diff --git a/tools/gg/graphGen.ml b/tools/gg/graphGen.ml index d01f2d1cf29636e0efe8883b0720be6c45821282..998b869559fa13a1d18e3dc2250ff04a6a145af6 100644 --- a/tools/gg/graphGen.ml +++ b/tools/gg/graphGen.ml @@ -10,7 +10,9 @@ open Sasacore exception Incorrect_attribute let min_max = ref None -let connected_cyclic = ref None +let connected = ref None +let cyclic = ref None +let tree = ref None let height:int option ref = ref None let generate_du_dur graph plan_udg t : unit = @@ -103,32 +105,34 @@ let all_attr : (Topology.t -> (string * string) list) = snd x) | Some x -> snd x); "is_connected", string_of_bool ( - match !connected_cyclic with + match !connected with | None -> ( Printf.eprintf "Computing the connection...\n"; flush stderr; - let x = Topology.is_connected_and_cyclic g in - connected_cyclic := Some x; - fst x) - | Some x -> fst x ); + let x = Topology.is_connected g in + connected := Some x; + x) + | Some x -> x ); "is_cyclic", string_of_bool - (match !connected_cyclic with + (match !cyclic with | None -> ( Printf.eprintf "Computing the cyclicity...\n"; flush stderr; - let x = Topology.is_connected_and_cyclic g in - connected_cyclic := Some x; - snd x) - | Some x -> snd x ); + let x = Topology.is_cyclic g in + cyclic := Some x; + x) + | Some x -> x ); "is_tree", string_of_bool - (match !connected_cyclic with + (match !tree with | None -> ( Printf.eprintf "Computing the tree-ness...\n"; flush stderr; - let x = Topology.is_connected_and_cyclic g in - connected_cyclic := Some x; - (fst x) && not (snd x)) - | Some x -> (fst x) && (snd x) ); + let x = Topology.is_tree g in + tree := Some x; + x + ) + | Some x -> x + ); "links_number", string_of_int ( Printf.eprintf "Computing the link_number...\n"; flush stderr; @@ -263,7 +267,7 @@ let () = ( | _ -> (Printf.fprintf stderr "Unexpected outcome. Command line : %s\n" (String.concat " " (Array.to_list Sys.argv)); assert false) in - if t.connected && not (fst (Topology.is_connected_and_cyclic g)) then + if t.connected && not (Topology.is_connected g) then if !trials > max_trials then None else (incr trials;gen_graph ()) else Some g