From 008ce5ac80cd545de45d9cbf18a23995af341c3e Mon Sep 17 00:00:00 2001 From: Thibault Martin <thi.martin.pro@pm.me> Date: Tue, 6 Aug 2024 12:50:42 +0200 Subject: [PATCH] [tests] Better tests for nearest functions --- tests/misc/oracle/test_dominators.res.oracle | 135 +------------------ tests/misc/test_dominators.ml | 99 ++++++++------ 2 files changed, 57 insertions(+), 177 deletions(-) diff --git a/tests/misc/oracle/test_dominators.res.oracle b/tests/misc/oracle/test_dominators.res.oracle index 83659d4eb8..76bebfc2cd 100644 --- a/tests/misc/oracle/test_dominators.res.oracle +++ b/tests/misc/oracle/test_dominators.res.oracle @@ -1,4 +1,5 @@ [kernel] Parsing test_dominators.c (with preprocessing) + Computing for function f: { int __retres; @@ -30,28 +31,6 @@ Computing for function f: Immediate dominators of f (sid, idom, ipostdom): (1, none, 3), (3, 1, 6), (4, 3, none), (5, 4, none), (6, 3, 53), (53, 6, 8), (8, 53, 54), (54, 8, 55), (55, 54, none) -Nearest common ancestors/child of f (sid, sid, ancestor, child): - (1, 1, none, 3), (1, 3, none, 6), (1, 4, none, none), (1, 5, none, none), - (1, 6, none, 53), (1, 53, none, 8), (1, 8, none, 54), (1, 54, none, 55), - (1, 55, none, none), (3, 1, none, 6), (3, 3, 1, 6), (3, 4, 1, none), - (3, 5, 1, none), (3, 6, 1, 53), (3, 53, 1, 8), (3, 8, 1, 54), - (3, 54, 1, 55), (3, 55, 1, none), (4, 1, none, none), (4, 3, 1, none), - (4, 4, 3, none), (4, 5, 3, none), (4, 6, 3, none), (4, 53, 3, none), - (4, 8, 3, none), (4, 54, 3, none), (4, 55, 3, none), (5, 1, none, none), - (5, 3, 1, none), (5, 4, 3, none), (5, 5, 4, none), (5, 6, 3, none), - (5, 53, 3, none), (5, 8, 3, none), (5, 54, 3, none), (5, 55, 3, none), - (6, 1, none, 53), (6, 3, 1, 53), (6, 4, 3, none), (6, 5, 3, none), - (6, 6, 3, 53), (6, 53, 3, 8), (6, 8, 3, 54), (6, 54, 3, 55), - (6, 55, 3, none), (53, 1, none, 8), (53, 3, 1, 8), (53, 4, 3, none), - (53, 5, 3, none), (53, 6, 3, 8), (53, 53, 6, 8), (53, 8, 6, 54), - (53, 54, 6, 55), (53, 55, 6, none), (8, 1, none, 54), (8, 3, 1, 54), - (8, 4, 3, none), (8, 5, 3, none), (8, 6, 3, 54), (8, 53, 6, 54), - (8, 8, 53, 54), (8, 54, 53, 55), (8, 55, 53, none), (54, 1, none, 55), - (54, 3, 1, 55), (54, 4, 3, none), (54, 5, 3, none), (54, 6, 3, 55), - (54, 53, 6, 55), (54, 8, 53, 55), (54, 54, 8, 55), (54, 55, 8, none), - (55, 1, none, none), (55, 3, 1, none), (55, 4, 3, none), (55, 5, 3, none), - (55, 6, 3, none), (55, 53, 6, none), (55, 8, 53, none), (55, 54, 8, none), - (55, 55, 54, none) Computing for function g: { @@ -89,41 +68,6 @@ Immediate dominators of g (sid, idom, ipostdom): (10, none, 12), (12, 10, 59), (13, 12, 15), (14, none, 15), (15, 13, 20), (16, 12, 57), (57, 16, 18), (18, 57, 58), (58, 18, 59), (20, 15, 59), (59, 12, none) -Nearest common ancestors/child of g (sid, sid, ancestor, child): - (10, 10, none, 12), (10, 12, none, 59), (10, 13, none, 59), - (10, 14, none, 59), (10, 15, none, 59), (10, 16, none, 59), - (10, 57, none, 59), (10, 18, none, 59), (10, 58, none, 59), - (10, 20, none, 59), (10, 59, none, none), (12, 10, none, 59), - (12, 12, 10, 59), (12, 13, 10, 59), (12, 14, none, 59), (12, 15, 10, 59), - (12, 16, 10, 59), (12, 57, 10, 59), (12, 18, 10, 59), (12, 58, 10, 59), - (12, 20, 10, 59), (12, 59, 10, none), (13, 10, none, 59), (13, 12, 10, 59), - (13, 13, 12, 15), (13, 14, none, 15), (13, 15, 12, 20), (13, 16, 12, 59), - (13, 57, 12, 59), (13, 18, 12, 59), (13, 58, 12, 59), (13, 20, 12, 59), - (13, 59, 12, none), (14, 10, none, 59), (14, 12, none, 59), - (14, 13, none, 15), (14, 14, none, 15), (14, 15, none, 20), - (14, 16, none, 59), (14, 57, none, 59), (14, 18, none, 59), - (14, 58, none, 59), (14, 20, none, 59), (14, 59, none, none), - (15, 10, none, 59), (15, 12, 10, 59), (15, 13, 12, 20), (15, 14, none, 20), - (15, 15, 13, 20), (15, 16, 12, 59), (15, 57, 12, 59), (15, 18, 12, 59), - (15, 58, 12, 59), (15, 20, 13, 59), (15, 59, 12, none), (16, 10, none, 59), - (16, 12, 10, 59), (16, 13, 12, 59), (16, 14, none, 59), (16, 15, 12, 59), - (16, 16, 12, 57), (16, 57, 12, 18), (16, 18, 12, 58), (16, 58, 12, 59), - (16, 20, 12, 59), (16, 59, 12, none), (57, 10, none, 59), (57, 12, 10, 59), - (57, 13, 12, 59), (57, 14, none, 59), (57, 15, 12, 59), (57, 16, 12, 18), - (57, 57, 16, 18), (57, 18, 16, 58), (57, 58, 16, 59), (57, 20, 12, 59), - (57, 59, 12, none), (18, 10, none, 59), (18, 12, 10, 59), (18, 13, 12, 59), - (18, 14, none, 59), (18, 15, 12, 59), (18, 16, 12, 58), (18, 57, 16, 58), - (18, 18, 57, 58), (18, 58, 57, 59), (18, 20, 12, 59), (18, 59, 12, none), - (58, 10, none, 59), (58, 12, 10, 59), (58, 13, 12, 59), (58, 14, none, 59), - (58, 15, 12, 59), (58, 16, 12, 59), (58, 57, 16, 59), (58, 18, 57, 59), - (58, 58, 18, 59), (58, 20, 12, 59), (58, 59, 12, none), (20, 10, none, 59), - (20, 12, 10, 59), (20, 13, 12, 59), (20, 14, none, 59), (20, 15, 13, 59), - (20, 16, 12, 59), (20, 57, 12, 59), (20, 18, 12, 59), (20, 58, 12, 59), - (20, 20, 15, 59), (20, 59, 12, none), (59, 10, none, none), - (59, 12, 10, none), (59, 13, 12, none), (59, 14, none, none), - (59, 15, 12, none), (59, 16, 12, none), (59, 57, 12, none), - (59, 18, 12, none), (59, 58, 12, none), (59, 20, 12, none), - (59, 59, 12, none) Computing for function h: { @@ -167,48 +111,6 @@ Immediate dominators of h (sid, idom, ipostdom): (22, none, none), (23, 22, none), (24, 23, none), (26, 24, none), (27, 26, none), (28, 26, none), (30, 28, none), (31, 30, none), (33, 30, none), (34, 27, none), (36, none, none) -Nearest common ancestors/child of h (sid, sid, ancestor, child): - (22, 22, none, none), (22, 23, none, none), (22, 24, none, none), - (22, 26, none, none), (22, 27, none, none), (22, 28, none, none), - (22, 30, none, none), (22, 31, none, none), (22, 33, none, none), - (22, 34, none, none), (22, 36, none, none), (23, 22, none, none), - (23, 23, 22, none), (23, 24, 22, none), (23, 26, 22, none), - (23, 27, 22, none), (23, 28, 22, none), (23, 30, 22, none), - (23, 31, 22, none), (23, 33, 22, none), (23, 34, 22, none), - (23, 36, none, none), (24, 22, none, none), (24, 23, 22, none), - (24, 24, 23, none), (24, 26, 23, none), (24, 27, 23, none), - (24, 28, 23, none), (24, 30, 23, none), (24, 31, 23, none), - (24, 33, 23, none), (24, 34, 23, none), (24, 36, none, none), - (26, 22, none, none), (26, 23, 22, none), (26, 24, 23, none), - (26, 26, 24, none), (26, 27, 24, none), (26, 28, 24, none), - (26, 30, 24, none), (26, 31, 24, none), (26, 33, 24, none), - (26, 34, 24, none), (26, 36, none, none), (27, 22, none, none), - (27, 23, 22, none), (27, 24, 23, none), (27, 26, 24, none), - (27, 27, 26, none), (27, 28, 26, none), (27, 30, 26, none), - (27, 31, 26, none), (27, 33, 26, none), (27, 34, 26, none), - (27, 36, none, none), (28, 22, none, none), (28, 23, 22, none), - (28, 24, 23, none), (28, 26, 24, none), (28, 27, 26, none), - (28, 28, 26, none), (28, 30, 26, none), (28, 31, 26, none), - (28, 33, 26, none), (28, 34, 26, none), (28, 36, none, none), - (30, 22, none, none), (30, 23, 22, none), (30, 24, 23, none), - (30, 26, 24, none), (30, 27, 26, none), (30, 28, 26, none), - (30, 30, 28, none), (30, 31, 28, none), (30, 33, 28, none), - (30, 34, 26, none), (30, 36, none, none), (31, 22, none, none), - (31, 23, 22, none), (31, 24, 23, none), (31, 26, 24, none), - (31, 27, 26, none), (31, 28, 26, none), (31, 30, 28, none), - (31, 31, 30, none), (31, 33, 30, none), (31, 34, 26, none), - (31, 36, none, none), (33, 22, none, none), (33, 23, 22, none), - (33, 24, 23, none), (33, 26, 24, none), (33, 27, 26, none), - (33, 28, 26, none), (33, 30, 28, none), (33, 31, 30, none), - (33, 33, 30, none), (33, 34, 26, none), (33, 36, none, none), - (34, 22, none, none), (34, 23, 22, none), (34, 24, 23, none), - (34, 26, 24, none), (34, 27, 26, none), (34, 28, 26, none), - (34, 30, 26, none), (34, 31, 26, none), (34, 33, 26, none), - (34, 34, 27, none), (34, 36, none, none), (36, 22, none, none), - (36, 23, none, none), (36, 24, none, none), (36, 26, none, none), - (36, 27, none, none), (36, 28, none, none), (36, 30, none, none), - (36, 31, none, none), (36, 33, none, none), (36, 34, none, none), - (36, 36, none, none) Computing for function i: { @@ -251,40 +153,6 @@ Immediate dominators of i (sid, idom, ipostdom): (38, none, 40), (40, 38, 44), (41, 40, 50), (43, 40, 44), (44, 40, 46), (46, 44, 47), (47, 46, 62), (48, 46, 49), (49, 48, 50), (50, 40, 44), (62, 47, none) -Nearest common ancestors/child of i (sid, sid, ancestor, child): - (38, 38, none, 40), (38, 40, none, 44), (38, 41, none, 44), - (38, 43, none, 44), (38, 44, none, 46), (38, 46, none, 47), - (38, 47, none, 62), (38, 48, none, 44), (38, 49, none, 44), - (38, 50, none, 44), (38, 62, none, none), (40, 38, none, 44), - (40, 40, 38, 44), (40, 41, 38, 44), (40, 43, 38, 44), (40, 44, 38, 46), - (40, 46, 38, 47), (40, 47, 38, 62), (40, 48, 38, 44), (40, 49, 38, 44), - (40, 50, 38, 44), (40, 62, 38, none), (41, 38, none, 44), (41, 40, 38, 44), - (41, 41, 40, 50), (41, 43, 40, 44), (41, 44, 40, 46), (41, 46, 40, 47), - (41, 47, 40, 62), (41, 48, 40, 50), (41, 49, 40, 50), (41, 50, 40, 44), - (41, 62, 40, none), (43, 38, none, 44), (43, 40, 38, 44), (43, 41, 40, 44), - (43, 43, 40, 44), (43, 44, 40, 46), (43, 46, 40, 47), (43, 47, 40, 62), - (43, 48, 40, 44), (43, 49, 40, 44), (43, 50, 40, 44), (43, 62, 40, none), - (44, 38, none, 46), (44, 40, 38, 46), (44, 41, 40, 46), (44, 43, 40, 46), - (44, 44, 40, 46), (44, 46, 40, 47), (44, 47, 40, 62), (44, 48, 40, 46), - (44, 49, 40, 46), (44, 50, 40, 46), (44, 62, 40, none), (46, 38, none, 47), - (46, 40, 38, 47), (46, 41, 40, 47), (46, 43, 40, 47), (46, 44, 40, 47), - (46, 46, 44, 47), (46, 47, 44, 62), (46, 48, 44, 47), (46, 49, 44, 47), - (46, 50, 40, 47), (46, 62, 44, none), (47, 38, none, 62), (47, 40, 38, 62), - (47, 41, 40, 62), (47, 43, 40, 62), (47, 44, 40, 62), (47, 46, 44, 62), - (47, 47, 46, 62), (47, 48, 46, 62), (47, 49, 46, 62), (47, 50, 40, 62), - (47, 62, 46, none), (48, 38, none, 44), (48, 40, 38, 44), (48, 41, 40, 50), - (48, 43, 40, 44), (48, 44, 40, 46), (48, 46, 44, 47), (48, 47, 46, 62), - (48, 48, 46, 49), (48, 49, 46, 50), (48, 50, 40, 44), (48, 62, 46, none), - (49, 38, none, 44), (49, 40, 38, 44), (49, 41, 40, 50), (49, 43, 40, 44), - (49, 44, 40, 46), (49, 46, 44, 47), (49, 47, 46, 62), (49, 48, 46, 50), - (49, 49, 48, 50), (49, 50, 40, 44), (49, 62, 46, none), (50, 38, none, 44), - (50, 40, 38, 44), (50, 41, 40, 44), (50, 43, 40, 44), (50, 44, 40, 46), - (50, 46, 40, 47), (50, 47, 40, 62), (50, 48, 40, 44), (50, 49, 40, 44), - (50, 50, 40, 44), (50, 62, 40, none), (62, 38, none, none), - (62, 40, 38, none), (62, 41, 40, none), (62, 43, 40, none), - (62, 44, 40, none), (62, 46, 44, none), (62, 47, 46, none), - (62, 48, 46, none), (62, 49, 46, none), (62, 50, 40, none), - (62, 62, 47, none) Dominators analysis: Stmt:1 -> {1} @@ -382,6 +250,7 @@ Dominators analysis: Postominators analysis: Empty +Trigger some errors/warnings : [kernel] Failure: Statement -1 is not part of a function [kernel] Warning: No statement in function stop: Dominators analysis cannot be done [kernel] Warning: No statement in function stop: PostDominators analysis cannot be done diff --git a/tests/misc/test_dominators.ml b/tests/misc/test_dominators.ml index c9c03a2b73..eb26706d80 100644 --- a/tests/misc/test_dominators.ml +++ b/tests/misc/test_dominators.ml @@ -9,9 +9,6 @@ let get = function let pp_res = Pretty_utils.pp_list ~pre:"(" ~sep:", " ~suf:")" Format.pp_print_string -let first_stmt_f = ref None -let first_stmt_g = ref None - (** For each statement of [f], find its immediate dominator and postdominator and print the triplets. *) let print_immediate f = @@ -26,24 +23,6 @@ let print_immediate f = f.svar.vname (Pretty_utils.pp_list ~pre:"@[" ~sep:",@ " ~suf:"@]" pp_res) res -(** For each couple of statement of [f], find their common ancestor and child - and print the quadruplets. *) -let print_nearest f = - assert (Dominators.nearest_common_ancestor [] = None); - let res = - List.map (fun s -> - List.map (fun s' -> - let dom = Dominators.nearest_common_ancestor [s; s'] in - let postdom = Dominators.nearest_common_child [s; s'] in - [string_of_int s.sid; string_of_int s'.sid; get dom; get postdom] - ) f.sallstmts - ) f.sallstmts - |> List.flatten - in - Format.printf "@[<v2>Nearest common ancestors/child of %s (sid, sid, ancestor, child):@;%a@]@;" - f.svar.vname - (Pretty_utils.pp_list ~pre:"@[" ~sep:",@ " ~suf:"@]" pp_res) res - (* Make sure that the difference between (post)dominators and strict (post)dominators of a statement [s] is equal to the singleton [s]. *) let test_strict f = @@ -63,12 +42,46 @@ let test_strict f = assert (Dominators.strictly_postdominates s s = false) ) f.sallstmts +let test_nearest kf f = + (* Simple test for empty list coverage of nearest function. *) + assert (Dominators.nearest_common_ancestor [] = None); + assert (Dominators.nearest_common_child [] = None); + + (* Since first and last statement do not have an ancestor/child, + common nearest for all statements should be empty. *) + assert (Dominators.nearest_common_ancestor f.sallstmts = None); + assert (Dominators.nearest_common_child f.sallstmts = None); + + if List.length f.sallstmts > 1 then begin + + let first = Kernel_function.find_first_stmt kf in + let last = Kernel_function.find_return kf in + let all_but_first = List.filter (fun s -> s != first) f.sallstmts in + let all_but_last = List.filter (fun s -> s != last) f.sallstmts in + + (* Test that all statements (except the first one) have as common ancestor + the first statement of f, unless one of them is unreachable. *) + begin match Dominators.nearest_common_ancestor all_but_first with + | None -> (* At leats one statement is unreachable. *) () + | Some ancestor -> assert (first == ancestor) + end; + + (* Test that all statements (except the last one) have as common child + the return statement of f, unless one of them is unreachable. This test + supposes that frama-c's normalization removed all but one return statement. + *) + begin match Dominators.nearest_common_child all_but_last with + | None -> (* At leats one statement is unreachable. *) () + | Some child -> assert (last == child) + end + end + class visitPostDom = object(self) inherit Visitor.frama_c_inplace method! vfunc f = let kf = Option.get (self#current_kf) in - Format.printf "@[<v>Computing for function %s:@;%a@?@;@?" + Format.printf "@.@[<v>Computing for function %s:@;%a@?@;@?@]" f.svar.vname Cil_printer.pp_block f.sbody; Dominators.compute_dominators kf; @@ -78,37 +91,35 @@ class visitPostDom = object(self) Dominators.print_dot_postdominators "postdom_graph" kf; print_immediate f; - print_nearest f; test_strict f; + test_nearest kf f; - Format.printf "@]@."; SkipChildren end -let catch f x = - try ignore (f x) - with Log.AbortFatal _ -> () - let cover_errors () = + Format.printf "Trigger some errors/warnings :@."; + + (* Test when a statement is not in a function. *) let loc = Cil_datatype.Location.unknown in - (* Test statement not in a function. *) let skip = Cil_builder.Pure.(cil_stmt ~loc skip) in - catch Dominators.get_postdominators skip; - let trigger kf = - match kf.fundec with - | Definition _ -> () - | Declaration _ -> - (* Test Kernel_function.find_first_stmt on decl. *) - Dominators.compute_dominators kf; - (* Test Kernel_function.find_return on decl. *) - Dominators.compute_postdominators kf; - (* Test print_dot on decl. *) - Dominators.print_dot_dominators "tmp" kf - in - Globals.Functions.iter trigger + try ignore (Dominators.get_postdominators skip) with Log.AbortFatal _ -> (); + + let trigger kf = + match kf.fundec with + | Definition _ -> () + | Declaration _ -> + (* Test Kernel_function.find_first_stmt on decl. *) + ignore @@ Dominators.compute_dominators kf; + (* Test Kernel_function.find_return on decl. *) + ignore @@ Dominators.compute_postdominators kf; + (* Test print_dot on decl. *) + ignore @@ Dominators.print_dot_dominators "tmp" kf + in + Globals.Functions.iter trigger let pretty () = - Format.printf "@[<v2>Dominators analysis:@;%a@]\n@." + Format.printf "@.@[<v2>Dominators analysis:@;%a@]\n@." Dominators.pretty_dominators (); Format.printf "@[<v2>Postominators analysis:@;%a@]\n@." Dominators.pretty_postdominators () @@ -117,7 +128,7 @@ let startup () = ignore (Cil.visitCilFileSameGlobals (new visitPostDom:>Cil.cilVisitor) (Ast.get ())); pretty (); Ast.mark_as_changed (); - Format.printf "Invalidate tables, which should now be empty\n@."; + Format.printf "Invalidate tables, which should now be empty\n"; pretty (); cover_errors () -- GitLab