Skip to content
Snippets Groups Projects
Commit 3dab330a authored by David Bühler's avatar David Bühler
Browse files

[kernel] Dominators: removes module StmtSetOpt and simply uses the empty set.

parent 40b60546
No related branches found
No related tags found
No related merge requests found
...@@ -45,34 +45,9 @@ module StmtSet = struct ...@@ -45,34 +45,9 @@ module StmtSet = struct
List.find_opt f' (elements set) List.find_opt f' (elements set)
end end
(* Used to store the result of the analysis. [None] means the statement is
unreachable while [Some set] means the statement was reached with this set
of statements. *)
module StmtSetOpt = struct
include Datatype.Option (StmtSet)
let remove s setopt =
Option.map (StmtSet.remove s) setopt
let mem s setopt =
Option.fold ~none:false ~some:(StmtSet.mem s) setopt
let inter s s' =
match s, s' with
| None, None -> None
| Some s, None | None, Some s -> Some s
| Some s, Some s' -> Some (StmtSet.inter s s')
let find_opt_for_all f setopt =
Option.bind setopt (StmtSet.find_opt_for_all f)
let pretty fmt setopt =
Pretty_utils.pp_opt ~none:"Top" StmtSet.pretty fmt setopt
end
module DotGraph = Graph.Graphviz.Dot ( module DotGraph = Graph.Graphviz.Dot (
struct struct
type t = string * (StmtSetOpt.t StmtTbl.t) type t = string * (StmtSet.t StmtTbl.t)
module V = struct module V = struct
type t = stmt type t = stmt
let pretty fmt v = Cil_printer.pp_stmt fmt v let pretty fmt v = Cil_printer.pp_stmt fmt v
...@@ -87,11 +62,7 @@ module DotGraph = Graph.Graphviz.Dot ( ...@@ -87,11 +62,7 @@ module DotGraph = Graph.Graphviz.Dot (
StmtTbl.iter (fun stmt _ -> f stmt) graph StmtTbl.iter (fun stmt _ -> f stmt) graph
let iter_edges_e f (_, graph) = let iter_edges_e f (_, graph) =
let do_edges stmt set_opt = StmtTbl.iter (fun stmt -> StmtSet.iter (fun p -> f (p, stmt))) graph
let do_edge p = f (p, stmt) in
Option.iter (fun set -> StmtSet.iter do_edge set) set_opt
in
StmtTbl.iter do_edges graph
let graph_attributes (title, _) = [`Label title] let graph_attributes (title, _) = [`Label title]
...@@ -145,7 +116,7 @@ module Compute (Analysis : Analysis) = struct ...@@ -145,7 +116,7 @@ module Compute (Analysis : Analysis) = struct
module Table = module Table =
Cil_state_builder.Stmt_hashtbl Cil_state_builder.Stmt_hashtbl
(StmtSetOpt) (StmtSet)
(struct (struct
let name = Analysis.name ^ "_table" let name = Analysis.name ^ "_table"
let dependencies = [Ast.self] let dependencies = [Ast.self]
...@@ -166,11 +137,11 @@ module Compute (Analysis : Analysis) = struct ...@@ -166,11 +137,11 @@ module Compute (Analysis : Analysis) = struct
(* Compute the analysis, initial state is empty. *) (* Compute the analysis, initial state is empty. *)
let result = Analysis.fixpoint kf StmtSet.empty in let result = Analysis.fixpoint kf StmtSet.empty in
(* Fill table with all statements. *) (* Fill table with all statements. *)
List.iter (fun stmt -> Table.add stmt None) f.sallstmts; List.iter (fun stmt -> Table.add stmt StmtSet.empty) f.sallstmts;
(* Update the table with analysis results. *) (* Update the table with analysis results. *)
Analysis.Result.iter_stmt (fun s set -> Analysis.Result.iter_stmt (fun s set ->
(* A statement always (post)dominates itself, so we add it here. *) (* A statement always (post)dominates itself, so we add it here. *)
Table.replace s (Some (StmtSet.add s set)) Table.replace s (StmtSet.add s set)
) result; ) result;
Kernel.feedback ~level:2 "done for function %a" Kernel.feedback ~level:2 "done for function %a"
Kernel_function.pretty kf Kernel_function.pretty kf
...@@ -190,53 +161,46 @@ module Compute (Analysis : Analysis) = struct ...@@ -190,53 +161,46 @@ module Compute (Analysis : Analysis) = struct
| Some v -> v | Some v -> v
(* Generic function to get the set of strict (post)dominators of [stmt]. *) (* Generic function to get the set of strict (post)dominators of [stmt]. *)
let get_strict stmt = get stmt |> StmtSetOpt.remove stmt let get_strict stmt = get stmt |> StmtSet.remove stmt
(* Generic function to test the (post)domination of 2 statements. *) (* Generic function to test the (post)domination of 2 statements. *)
let mem a b = get b |> StmtSetOpt.mem a let mem a b = get b |> StmtSet.mem a
(* Generic function to test the strict (post)domination of 2 statements. *) (* Generic function to test the strict (post)domination of 2 statements. *)
let mem_strict a b = get_strict b |> StmtSetOpt.mem a let mem_strict a b = get_strict b |> StmtSet.mem a
(* The nearest common ancestor (resp. children) is the ancestor which is (* The nearest common ancestor (resp. children) is the ancestor which is
dominated (resp. postdominated) by all common ancestors, ie. the lowest dominated (resp. postdominated) by all common ancestors, ie. the lowest
(resp. highest) ancestor in the domination tree. *) (resp. highest) ancestor in the domination tree. *)
let nearest stmtl = let nearest stmtl =
let exception Unreachable in
(* Get the set of strict (post)doms for each statement and intersect them to (* Get the set of strict (post)doms for each statement and intersect them to
keep the common ones. If one of them is None (unreachable), they do not keep the common ones. If one of them is unreachable, they do not
share a common ancestor/children. *) share a common ancestor/children. *)
let common_set = let common_set =
try match stmtl with
List.fold_left (fun acc s -> | [] -> StmtSet.empty
match get_strict s with | stmt :: tail ->
| None -> raise Unreachable List.fold_left
| set -> StmtSetOpt.inter acc set (fun acc s -> StmtSet.inter acc (get_strict s))
) None stmtl (get_strict stmt) tail
with Unreachable -> None
in in
(* Try to find a statement [s] in [common_set] which is (post)dominated by (* Try to find a statement [s] in [common_set] which is (post)dominated by
all statements of the [common_set]. *) all statements of the [common_set]. *)
StmtSetOpt.find_opt_for_all (Fun.flip mem) common_set StmtSet.find_opt_for_all (Fun.flip mem) common_set
let pretty fmt () = let pretty fmt () =
let l = Table.to_seq () |> List.of_seq in let l = Table.to_seq () |> List.of_seq in
Pretty_utils.pp_list ~pre:"@[<v>" ~sep:"@;" ~empty:"Empty" Pretty_utils.pp_list ~pre:"@[<v>" ~sep:"@;" ~empty:"Empty"
(fun fmt (k,v) -> (fun fmt (k,v) ->
Format.fprintf fmt "Stmt:%d -> @[%a@]" k.sid StmtSetOpt.pretty v Format.fprintf fmt "Stmt:%d -> @[%a@]" k.sid StmtSet.pretty v
) fmt l ) fmt l
let get_set graph stmt = let get_set graph stmt =
match StmtTbl.find_opt graph stmt with match StmtTbl.find_opt graph stmt with
| Some None -> assert false | Some l -> l
| Some (Some l) -> l
| None -> | None ->
match get_strict stmt with let set = get_strict stmt in
| Some set -> StmtTbl.add graph stmt set; set
StmtTbl.add graph stmt (Some set); set
| None ->
StmtTbl.add graph stmt None;
raise Not_found
(* [s_set] are [s] (post)dominators, including [s]. We don't have to represent (* [s_set] are [s] (post)dominators, including [s]. We don't have to represent
the relation between [s] and [s], so [get_set] removes it. And because the the relation between [s] and [s], so [get_set] removes it. And because the
...@@ -246,12 +210,10 @@ module Compute (Analysis : Analysis) = struct ...@@ -246,12 +210,10 @@ module Compute (Analysis : Analysis) = struct
let reduce graph s = let reduce graph s =
(* Union of all (post)dominators of [s] (post)dominators [s_set]. *) (* Union of all (post)dominators of [s] (post)dominators [s_set]. *)
let unions p acc = get_set graph p |> StmtSet.union acc in let unions p acc = get_set graph p |> StmtSet.union acc in
try let s_set = get_set graph s in
let s_set = get_set graph s in let p_sets = StmtSet.fold unions s_set StmtSet.empty in
let p_sets = StmtSet.fold unions s_set StmtSet.empty in let res = StmtSet.diff s_set p_sets in
let res = StmtSet.diff s_set p_sets in StmtTbl.replace graph s res
StmtTbl.replace graph s (Some res)
with Not_found -> ()
let build_dot filename kf = let build_dot filename kf =
match kf.fundec with match kf.fundec with
......
...@@ -56,24 +56,24 @@ val compute_postdominators : kernel_function -> unit ...@@ -56,24 +56,24 @@ val compute_postdominators : kernel_function -> unit
@since Frama-C+dev @since Frama-C+dev
*) *)
val get_dominators : stmt -> Cil_datatype.Stmt.Hptset.t option val get_dominators : stmt -> Cil_datatype.Stmt.Hptset.t
(** Return the [set] of dominators of the given statement. [None] means the (** Return the [set] of dominators of the given statement.
statement is unreachable. The empty set means the statement is unreachable.
@since Frama-C+dev @since Frama-C+dev
*) *)
val get_postdominators : stmt -> Cil_datatype.Stmt.Hptset.t option val get_postdominators : stmt -> Cil_datatype.Stmt.Hptset.t
(** Return the [set] of postdominators of the given statement. [None] means the (** Return the [set] of postdominators of the given statement.
statement is unreachable. The empty set means the statement is unreachable.
@since Frama-C+dev @since Frama-C+dev
*) *)
val get_strict_dominators : stmt -> Cil_datatype.Stmt.Hptset.t option val get_strict_dominators : stmt -> Cil_datatype.Stmt.Hptset.t
(** Same as [get_dominators] but exclude the statement itself. (** Same as [get_dominators] but exclude the statement itself.
@since Frama-C+dev @since Frama-C+dev
*) *)
val get_strict_postdominators : stmt -> Cil_datatype.Stmt.Hptset.t option val get_strict_postdominators : stmt -> Cil_datatype.Stmt.Hptset.t
(** Same as [get_postdominators] but exclude the statement itself. (** Same as [get_postdominators] but exclude the statement itself.
@since Frama-C+dev @since Frama-C+dev
*) *)
......
...@@ -296,7 +296,7 @@ Dominators analysis: ...@@ -296,7 +296,7 @@ Dominators analysis:
Stmt:10 -> {10} Stmt:10 -> {10}
Stmt:12 -> {10,12} Stmt:12 -> {10,12}
Stmt:13 -> {10,12,13} Stmt:13 -> {10,12,13}
Stmt:14 -> Top Stmt:14 -> {}
Stmt:15 -> {10,12,13,15} Stmt:15 -> {10,12,13,15}
Stmt:16 -> {10,12,16} Stmt:16 -> {10,12,16}
Stmt:18 -> {10,12,16,18,57} Stmt:18 -> {10,12,16,18,57}
...@@ -311,7 +311,7 @@ Dominators analysis: ...@@ -311,7 +311,7 @@ Dominators analysis:
Stmt:31 -> {22,23,24,26,28,30,31} Stmt:31 -> {22,23,24,26,28,30,31}
Stmt:33 -> {22,23,24,26,28,30,33} Stmt:33 -> {22,23,24,26,28,30,33}
Stmt:34 -> {22,23,24,26,27,34} Stmt:34 -> {22,23,24,26,27,34}
Stmt:36 -> Top Stmt:36 -> {}
Stmt:38 -> {38} Stmt:38 -> {38}
Stmt:40 -> {38,40} Stmt:40 -> {38,40}
Stmt:41 -> {38,40,41} Stmt:41 -> {38,40,41}
...@@ -333,8 +333,8 @@ Dominators analysis: ...@@ -333,8 +333,8 @@ Dominators analysis:
Postominators analysis: Postominators analysis:
Stmt:1 -> {1,3,6,8,53,54,55} Stmt:1 -> {1,3,6,8,53,54,55}
Stmt:3 -> {3,6,8,53,54,55} Stmt:3 -> {3,6,8,53,54,55}
Stmt:4 -> Top Stmt:4 -> {}
Stmt:5 -> Top Stmt:5 -> {}
Stmt:6 -> {6,8,53,54,55} Stmt:6 -> {6,8,53,54,55}
Stmt:8 -> {8,54,55} Stmt:8 -> {8,54,55}
Stmt:10 -> {10,12,59} Stmt:10 -> {10,12,59}
...@@ -345,16 +345,16 @@ Postominators analysis: ...@@ -345,16 +345,16 @@ Postominators analysis:
Stmt:16 -> {16,18,57,58,59} Stmt:16 -> {16,18,57,58,59}
Stmt:18 -> {18,58,59} Stmt:18 -> {18,58,59}
Stmt:20 -> {20,59} Stmt:20 -> {20,59}
Stmt:22 -> Top Stmt:22 -> {}
Stmt:23 -> Top Stmt:23 -> {}
Stmt:24 -> Top Stmt:24 -> {}
Stmt:26 -> Top Stmt:26 -> {}
Stmt:27 -> Top Stmt:27 -> {}
Stmt:28 -> Top Stmt:28 -> {}
Stmt:30 -> Top Stmt:30 -> {}
Stmt:31 -> Top Stmt:31 -> {}
Stmt:33 -> Top Stmt:33 -> {}
Stmt:34 -> Top Stmt:34 -> {}
Stmt:36 -> {36} Stmt:36 -> {36}
Stmt:38 -> {38,40,44,46,47,62} Stmt:38 -> {38,40,44,46,47,62}
Stmt:40 -> {40,44,46,47,62} Stmt:40 -> {40,44,46,47,62}
......
...@@ -48,11 +48,9 @@ let print_nearest f = ...@@ -48,11 +48,9 @@ let print_nearest f =
(post)dominators of a statement [s] is equal to the singleton [s]. *) (post)dominators of a statement [s] is equal to the singleton [s]. *)
let test_strict f = let test_strict f =
let test s dom strict_dom = let test s dom strict_dom =
match dom, strict_dom with if StmtSet.is_empty dom
| Some dom, Some strict_dom -> then assert (StmtSet.is_empty strict_dom)
assert (StmtSet.diff dom strict_dom == StmtSet.singleton s) else assert (StmtSet.diff dom strict_dom == StmtSet.singleton s)
| None, None -> ()
| _ -> assert false
in in
List.iter (fun s -> List.iter (fun s ->
let dom = Dominators.get_dominators s in let dom = Dominators.get_dominators s in
......
0% Loading or .
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment