From 220c5cc2c2e1785848ac4377ca0a9e0ce55ccbd5 Mon Sep 17 00:00:00 2001 From: Erwan Jahier <erwan.jahier@univ-grenoble-alpes.fr> Date: Tue, 4 May 2021 08:50:28 +0200 Subject: [PATCH] Chore: use a string map instead of an assoc list for getting pid's neighbors --- lib/sasa/sasaRun.ml | 29 +++++++----------- lib/sasacore/daemon.ml | 62 +++++++++++++++++++------------------- lib/sasacore/daemon.mli | 9 +++--- lib/sasacore/evil.ml | 30 +++++++++--------- lib/sasacore/evil.mli | 12 +++++--- lib/sasacore/simuState.ml | 38 ++++++++++++++++++----- lib/sasacore/simuState.mli | 14 ++++++--- src/sasaMain.ml | 46 +++++++++------------------- test/coloring/config.ml | 1 + 9 files changed, 123 insertions(+), 118 deletions(-) diff --git a/lib/sasa/sasaRun.ml b/lib/sasa/sasaRun.ml index f51feb52..5b2301cf 100644 --- a/lib/sasa/sasaRun.ml +++ b/lib/sasa/sasaRun.ml @@ -18,7 +18,7 @@ open Process let (from_sasa_env : 'v SimuState.t -> RdbgPlugin.sl) = fun st -> List.fold_left - (fun acc (p,_) -> + (fun acc p -> let state = Env.get st.config p.pid in let sl = SasaState.to_rdbg_subst p.pid state in acc@sl @@ -37,18 +37,14 @@ let (get_sl_out: bool -> 'v Process.t list -> bool list list -> RdbgPlugin.sl) = pl ll ) +module StringMap = Map.Make(String) let (compute_potentiel: 'v SimuState.t -> RdbgPlugin.sl) = fun st -> match Register.get_potential () with | None -> [] | Some user_pf -> - let pidl = List.map (fun (p,_) -> p.Process.pid) st.network in - let get_info pid = - let nl = snd (List.find (fun (p,_) -> p.Process.pid = pid) st.network) in - Env.get st.config pid, - List.map (fun n -> n, n.Register.pid) nl - in - let p = (user_pf pidl get_info) in + let pidl = List.map (fun p -> p.Process.pid) st.network in + let p = user_pf pidl (SimuState.neigbors_of_pid st) in [("potential", Data.F p)] let (compute_legitimate: bool -> 'v SimuState.t -> bool) = @@ -57,18 +53,13 @@ let (compute_legitimate: bool -> 'v SimuState.t -> bool) = match Register.get_legitimate () with | None -> silent | Some f -> - let pidl = List.map (fun (p,_) -> p.Process.pid) st.network in - let get_info pid = - let nl = snd (List.find (fun (p,_) -> p.Process.pid = pid) st.network) in - Env.get st.config pid, - List.map (fun n -> n, n.Register.pid) nl - in - f pidl get_info + let pidl = List.map (fun p -> p.Process.pid) st.network in + f pidl (SimuState.neigbors_of_pid st) open SimuState let (make_do: string array -> 'v SimuState.t -> RdbgPlugin.t) = fun argv st -> - let pl = fst (List.split st.network) in + let pl = st.network in let prog_id = Printf.sprintf "%s (with sasa Version %s)" (String.concat " " (Array.to_list argv)) SasaVersion.str in @@ -111,7 +102,8 @@ let (make_do: string array -> 'v SimuState.t -> RdbgPlugin.t) = (* if was_silent then failwith "Silent"; *) (* 2: read the actions from the outside process, i.e., from sl_in *) let _, pnal = Daemon.f st.sasarg.dummy_input - (st.sasarg.verbose > 0) st.sasarg.daemon st.network st.config pre_pnall pre_enab_ll + (st.sasarg.verbose > 0) st.sasarg.daemon st.network + (SimuState.neigbors_of_pid st) st.config pre_pnall pre_enab_ll (get_action_value sl_in) in (* 3: Do the steps *) @@ -148,7 +140,8 @@ let (make_do: string array -> 'v SimuState.t -> RdbgPlugin.t) = else (* 2: read the actions from the outside process, i.e., from sl_in *) let activate_val, pnal = Daemon.f st.sasarg.dummy_input - (st.sasarg.verbose > 0) st.sasarg.daemon st.network st.config pnall enab_ll + (st.sasarg.verbose > 0) st.sasarg.daemon st.network (SimuState.neigbors_of_pid st) + st.config pnall enab_ll (get_action_value sl_in) in (* 3: Do the steps *) diff --git a/lib/sasacore/daemon.ml b/lib/sasacore/daemon.ml index 9756932d..e4f2a3ec 100644 --- a/lib/sasacore/daemon.ml +++ b/lib/sasacore/daemon.ml @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 08/12/2020 (at 16:09) by Erwan Jahier> *) +(* Time-stamp: <modified the 03/05/2021 (at 16:16) by Erwan Jahier> *) type t = | Synchronous (* select all actions *) @@ -120,35 +120,35 @@ let (get_activate_val: 'v pna list -> 'v Process.t list -> bool list list)= let al = List.map (fun (p,_,a) -> p,a) al in List.map (List.map (fun a -> List.mem a al)) actions -let (f: bool -> bool -> t -> ('v Process.t * 'v Register.neighbor list) list -> - 'v Env.t -> 'v pna list list -> bool list list -> - (string -> string -> bool) -> bool list list * 'v pna list) = - fun dummy_input verbose_mode daemon p_nl_l e all enab get_action_value -> +let (f: bool -> bool -> t -> 'v Process.t list -> + (string -> 'v * ('v Register.neighbor * string) list) -> 'v Env.t -> + 'v pna list list -> bool list list -> (string -> string -> bool) -> + bool list list * 'v pna list) = + fun dummy_input verbose_mode daemon pl neigbors_of_pid e all enab get_action_value -> let nall = remove_empty_list all in if nall = [] then assert false (* failwith "Silent" *); - let pl = fst(List.split p_nl_l) in - if daemon <> Custom && dummy_input then - ignore (RifRead.bool verbose_mode ((List.hd pl).pid) ""); - match daemon with - | Synchronous -> - let al = synchrone nall in - get_activate_val al pl, al - | Central -> - let al = central nall in - get_activate_val al pl, al - | LocallyCentral -> - let al = locally_central nall in - get_activate_val al pl, al - | Distributed -> - let al = distributed nall in - get_activate_val al pl, al - | Greedy -> - let al = Evil.greedy verbose_mode e p_nl_l nall in - get_activate_val al pl, al - | GreedyCentral -> - let al = Evil.greedy_central verbose_mode e p_nl_l nall in - get_activate_val al pl, al - | Bad i -> - let al = Evil.bad i e nall in - get_activate_val al pl, al - | Custom -> custom all pl enab get_action_value + if daemon <> Custom && dummy_input then + ignore (RifRead.bool verbose_mode ((List.hd pl).pid) ""); + match daemon with + | Synchronous -> + let al = synchrone nall in + get_activate_val al pl, al + | Central -> + let al = central nall in + get_activate_val al pl, al + | LocallyCentral -> + let al = locally_central nall in + get_activate_val al pl, al + | Distributed -> + let al = distributed nall in + get_activate_val al pl, al + | Greedy -> + let al = Evil.greedy verbose_mode e pl neigbors_of_pid nall in + get_activate_val al pl, al + | GreedyCentral -> + let al = Evil.greedy_central verbose_mode e pl neigbors_of_pid nall in + get_activate_val al pl, al + | Bad i -> + let al = Evil.bad i e nall in + get_activate_val al pl, al + | Custom -> custom all pl enab get_action_value diff --git a/lib/sasacore/daemon.mli b/lib/sasacore/daemon.mli index 182fca66..1c5d94fb 100644 --- a/lib/sasacore/daemon.mli +++ b/lib/sasacore/daemon.mli @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 30/09/2020 (at 13:53) by Erwan Jahier> *) +(* Time-stamp: <modified the 03/05/2021 (at 16:16) by Erwan Jahier> *) type t = | Synchronous (* select all actions *) @@ -44,7 +44,8 @@ nb: it is possible that we read on stdin that an action should be inhibit the activation. *) -val f : bool -> bool -> t -> ('v Process.t * 'v Register.neighbor list) list -> - 'v Env.t -> 'v pna list list -> bool list list -> - (string -> string -> bool) -> bool list list * 'v pna list +val f : bool -> bool -> t -> 'v Process.t list -> + (string -> 'v * ('v Register.neighbor * string) list) -> + 'v Env.t -> 'v pna list list -> bool list list -> + (string -> string -> bool) -> bool list list * 'v pna list diff --git a/lib/sasacore/evil.ml b/lib/sasacore/evil.ml index 732e349c..e8ed51c1 100644 --- a/lib/sasacore/evil.ml +++ b/lib/sasacore/evil.ml @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 07/12/2020 (at 10:20) by Erwan Jahier> *) +(* Time-stamp: <modified the 03/05/2021 (at 16:17) by Erwan Jahier> *) type 'v pna = 'v Process.t * 'v Register.neighbor list * Register.action @@ -118,20 +118,20 @@ let time3 verb lbl f x y z = if verb then Printf.eprintf " [%s] Execution time: %fs\n" lbl (Sys.time() -. t); fxy -let (greedy: bool -> 'v Env.t -> ('v Process.t * 'v Register.neighbor list) list -> - 'v pna list list -> 'v pna list) = - fun verb e p_nl_l all -> +let (greedy: bool -> 'v Env.t -> 'v Process.t list -> + (string -> 'v * ('v Register.neighbor * string) list) -> 'v pna list list -> + 'v pna list) = + fun verb e pl neigbors_of_pid all -> assert (all<>[]); match Register.get_potential () with | None -> failwith "No potential function has been provided" | Some user_pf -> let pf pnal = (* pnal contains a list of activated processes *) - let pidl = List.map (fun (p,_) -> p.Process.pid) p_nl_l in + let pidl = List.map (fun p -> p.Process.pid) pl in let ne = Step.f pnal e in let get_info pid = - let nl = snd (List.find (fun (p,_) -> p.Process.pid = pid) p_nl_l) in - Env.get ne pid, - List.map (fun n -> n, n.Register.pid) nl + let _, nl = neigbors_of_pid pid in + Env.get ne pid, nl in user_pf pidl get_info in @@ -160,20 +160,20 @@ let (greedy: bool -> 'v Env.t -> ('v Process.t * 'v Register.neighbor list) list if verb then Printf.eprintf " [Evil.greedy] Number of trials: %i\n%!" !cpt; res -let (greedy_central: bool -> 'v Env.t -> ('v Process.t * 'v Register.neighbor list) list - -> 'v pna list list -> 'v pna list) = - fun verb e p_nl_l all -> +(* val greedy_central: bool -> 'v Env.t -> ('v Process.t * 'v Register.neighbor list) list -> *) +let (greedy_central: bool -> 'v Env.t -> 'v Process.t list -> + (string -> 'v * ('v Register.neighbor * string) list) -> 'v pna list list -> 'v pna list) = + fun verb e pl neigbors_of_pid all -> assert (all<>[]); match Register.get_potential () with | None -> failwith "No potential function has been provided" | Some user_pf -> let pf pna = - let pidl = List.map (fun (p,_) -> p.Process.pid) p_nl_l in + let pidl = List.map (fun p -> p.Process.pid) pl in let ne = Step.f [pna] e in let get_info pid = - let nl = snd (List.find (fun (p,_) -> p.Process.pid = pid) p_nl_l) in - Env.get ne pid, - List.map (fun n -> n, n.Register.pid) nl + let _, nl = neigbors_of_pid pid in + Env.get ne pid, nl in user_pf pidl get_info in diff --git a/lib/sasacore/evil.mli b/lib/sasacore/evil.mli index 10d6aa37..6cee4338 100644 --- a/lib/sasacore/evil.mli +++ b/lib/sasacore/evil.mli @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 07/12/2020 (at 10:16) by Erwan Jahier> *) +(* Time-stamp: <modified the 03/05/2021 (at 16:17) by Erwan Jahier> *) (** This module gathers daemons that tries to reach the worst case with @@ -6,15 +6,17 @@ type 'v pna = 'v Process.t * 'v Register.neighbor list * Register.action -(** [greedy verb e p_nl_l all] take the worst case among the combinations of +(** [greedy verb e pl neigbors_of_pid all] take the worst case among the combinations of length 1, i.e., O(2^n) where n is the number of enabled processes (|all|) *) -val greedy: bool -> 'v Env.t -> ('v Process.t * 'v Register.neighbor list) list -> +val greedy: bool -> 'v Env.t -> 'v Process.t list -> + (string -> 'v * ('v Register.neighbor * string) list) -> 'v pna list list -> 'v pna list (** Ditto, but for central daemons (of a connected component) *) -val greedy_central: bool -> 'v Env.t -> ('v Process.t * 'v Register.neighbor list) list -> - 'v pna list list -> 'v pna list +val greedy_central: bool -> 'v Env.t -> 'v Process.t list -> + (string -> 'v * ('v Register.neighbor * string) list) -> 'v pna list list -> + 'v pna list (** Returns the worst case among the combinations of length 1 for convex potential functions, and just a bad one otherwise (O(n) diff --git a/lib/sasacore/simuState.ml b/lib/sasacore/simuState.ml index 787df8f1..bd1d9875 100644 --- a/lib/sasacore/simuState.ml +++ b/lib/sasacore/simuState.ml @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 16/04/2021 (at 15:39) by Erwan Jahier> *) +(* Time-stamp: <modified the 04/05/2021 (at 08:36) by Erwan Jahier> *) open Register @@ -37,32 +37,49 @@ let (dump_process: string -> 'v Process.t * 'v Register.neighbor list -> unit) = open Process open SasArg +module StringMap = Map.Make(String) + type 'v t = { sasarg: SasArg.t; - network: ('v Process.t * 'v Register.neighbor list) list; + (* network: ('v Process.t * 'v Register.neighbor list) list; *) + network: 'v Process.t list; + neighbors: ('v Register.neighbor list) Map.Make(String).t; config: 'v Env.t } +let (neigbors_of_pid : 'v t -> pid -> 's * ('s neighbor * pid) list) = + fun st pid -> + let nl = + match StringMap.find_opt pid st.neighbors with + | Some x -> x + | None -> + failwith ( + Printf.sprintf "no %s found in %s" pid + (String.concat "," (List.map (fun p -> p.Process.pid) st.network))) + in + Env.get st.config pid, List.map (fun n -> n, n.Register.pid) nl + + let (update_neighbor_env: 'v Env.t -> 'v Register.neighbor list -> 'v Register.neighbor list) = fun e nl -> List.map (fun n -> { n with state = Env.get_copy e n.Register.pid }) nl -let update_network config network = List.map - (fun (p,nl) -> p, update_neighbor_env config nl) - network +let update_neighbors config neighbors = StringMap.map + (fun nl -> update_neighbor_env config nl) + neighbors let (update_config: 'v Env.t -> 'v t -> 'v t) = fun e st -> let verb = !Register.verbose_level > 0 in if verb then Printf.eprintf " ===> update_neighbor_env\n%!"; - { st with network = update_network e st.network ; config = e } + { st with neighbors = update_neighbors e st.neighbors ; config = e } type 'v enable_processes = ('v Process.t * 'v Register.neighbor list * Register.action) list list * bool list list let (get_enable_processes: 'v t -> 'v enable_processes) = fun st -> - let pl_n = st.network in + let pl_n = List.map (fun p -> p, StringMap.find p.pid st.neighbors) st.network in let e = st.config in assert (pl_n <> []); let all = List.fold_left @@ -306,9 +323,14 @@ let (make : bool -> string array -> 'v t) = Printf.eprintf "Ignoring the first vectors of sasa inputs\n%!"; ); if !Register.verbose_level > 0 then Printf.eprintf "==> SimuState.make done !\n%!"; + let neighbors = + List.fold_left (fun acc (p,nl) -> StringMap.add p.pid nl acc) + StringMap.empty pl_n + in { sasarg = args; - network = pl_n; + network = pl; + neighbors = neighbors; config = e } with diff --git a/lib/sasacore/simuState.mli b/lib/sasacore/simuState.mli index b7db9e31..6c02a144 100644 --- a/lib/sasacore/simuState.mli +++ b/lib/sasacore/simuState.mli @@ -1,4 +1,4 @@ -(* Time-stamp: <modified the 16/04/2021 (at 15:31) by Erwan Jahier> *) +(* Time-stamp: <modified the 04/05/2021 (at 08:27) by Erwan Jahier> *) (** The module is used by - the main sasa simulation loop (in ../../src/sasaMain.ml) @@ -6,10 +6,13 @@ *) + + (* type 'v t = SasArg.t * 'v layout * 'v Env.t *) type 'v t = { sasarg: SasArg.t; - network: ('v Process.t * 'v Register.neighbor list) list; + network: 'v Process.t list; + neighbors: ('v Register.neighbor list) Map.Make(String).t; (* pid's neigbors *) config: 'v Env.t } @@ -23,9 +26,10 @@ val get_enable_processes: 'v t -> 'v enable_processes (** update the config and network processes *) val update_config: 'v Env.t -> 'v t -> 'v t - + +(** Get pid's state and neigbors *) +val neigbors_of_pid : 'v t -> string -> 'v * ('v Register.neighbor * string) list + (* For SasaRun *) val get_inputs_rif_decl : SasArg.t -> 'v Process.t list -> (string * string) list val get_outputs_rif_decl: SasArg.t -> 'v Process.t list -> (string * string) list - - diff --git a/src/sasaMain.ml b/src/sasaMain.ml index b6137dad..65f1918a 100644 --- a/src/sasaMain.ml +++ b/src/sasaMain.ml @@ -55,28 +55,20 @@ let bool_ll_to_string bll = String.concat " " (List.map (fun b -> if b then "t" else "f") (List.flatten bll)) -let legitimate p_nl_l e = +open Sasacore.SimuState + +let legitimate st = match Register.get_legitimate () with | None -> false | Some ulf -> - let pidl = List.map (fun (p,_) -> p.Process.pid) p_nl_l in - let rec from_pid p_nl_l pid = (* XXX use StringMap instead *) - match p_nl_l with - | [] -> assert false (* sno *) - | (p,nl)::tail -> - if p.Process.pid = pid then - let nl = List.map (fun n -> n,n.Register.pid) nl in - Env.get e pid, - nl - else - from_pid tail pid - in - ulf pidl (from_pid p_nl_l) + let pidl = List.map (fun p -> p.Process.pid) st.network in + ulf pidl (SimuState.neigbors_of_pid st) -open Sasacore.SimuState +module StringMap = Map.Make(String) let inject_fault ff st = - let update_nodes e (p,nl) = + let update_nodes e p = + let nl = StringMap.find p.Process.pid st.neighbors in let pid = p.Process.pid in let v = Env.get e pid in let v = ff (List.length nl) pid v in @@ -92,19 +84,8 @@ let (compute_potentiel: 'v SimuState.t -> string) = match Register.get_potential () with | None -> "" | Some user_pf -> - let pidl = List.map (fun (p,_) -> p.Process.pid) st.network in - let get_info pid = - let nl = match List.find_opt (fun (p,_) -> p.Process.pid = pid) st.network with - None -> - failwith ( - Printf.sprintf "no %s found in %s" pid - (String.concat "," (List.map (fun (p,_) -> p.Process.pid) st.network))) - | Some (_,x) -> x - in - Env.get st.config pid, - List.map (fun n -> n, n.Register.pid) nl - in - let p = user_pf pidl get_info in + let pidl = List.map (fun p -> p.Process.pid) st.network in + let p = user_pf pidl (SimuState.neigbors_of_pid st) in string_of_float p @@ -115,7 +96,7 @@ let (simustep: int -> int -> string -> 'v SimuState.t -> 'v SimuState.t * string if verb then Printf.eprintf "==> SasaSimuState.simustep :1: Get enable processes\n%!"; let all, enab_ll = Sasacore.SimuState.get_enable_processes st in let pot = compute_potentiel st in - let pl = fst(List.split st.network) in + let pl = st.network in let st, all, enab_ll = if (* not (args.rif) && *) @@ -136,7 +117,7 @@ let (simustep: int -> int -> string -> 'v SimuState.t -> 'v SimuState.t * string let all, enab_ll = Sasacore.SimuState.get_enable_processes st in st, all, enab_ll ) - else if legitimate st.network st.config then ( + else if legitimate st then ( match Register.get_fault () with | None -> print_step n i pot st.sasarg st.config pl activate_val enab_ll; @@ -161,7 +142,8 @@ let (simustep: int -> int -> string -> 'v SimuState.t -> 'v SimuState.t * string if verb then Printf.eprintf "==> SasaSimuState.simustep : 2: read the actions\n%!"; let get_action_value = RifRead.bool (st.sasarg.verbose > 1) in let next_activate_val, pnal = Daemon.f st.sasarg.dummy_input - (st.sasarg.verbose >= 1) st.sasarg.daemon st.network st.config all enab_ll get_action_value + (st.sasarg.verbose >= 1) st.sasarg.daemon st.network (SimuState.neigbors_of_pid st) + st.config all enab_ll get_action_value in List.iter (List.iter (fun b -> if b then incr moves)) next_activate_val; update_round next_activate_val enab_ll; diff --git a/test/coloring/config.ml b/test/coloring/config.ml index aa5fd985..2c5a0137 100644 --- a/test/coloring/config.ml +++ b/test/coloring/config.ml @@ -10,5 +10,6 @@ let clash_number pidl get = float_of_int !clash let potential = Some clash_number +(* let potential = None *) let legitimate = None let fault = None -- GitLab