diff --git a/ptests/ptests.ml b/ptests/ptests.ml index c68ccb17823fe4fa0e1d5a329a78ad9a0a840073..4ef2fba4abd80b3686f241416a47894cbc518db9 100644 --- a/ptests/ptests.ml +++ b/ptests/ptests.ml @@ -20,10 +20,6 @@ (* *) (**************************************************************************) -(** the options to launch the toplevel with if the test file is not - annotated with test options *) -let default_options = "-journal-disable -check" - let system = if Sys.os_type = "Win32" then fun f -> @@ -172,6 +168,29 @@ let dir_config_file = "test_config" the pattern [test_file_regexp] will be considered as test files *) let test_file_regexp = ".*\\.\\(c\\|i\\)$" +(* Splits the command string to separate the command name from the parameters + [let cmd_name,param=command_partition cmd in assert cmd=cmd_name^param] +*) +let command_partition = + let regexp_unescaped_blank = Str.regexp "[^\\ ] " in + fun cmd -> + match str_bounded_full_split regexp_unescaped_blank cmd 2 with + | [ Str.Text cmd ] -> + cmd, "" + | [ Str.Text cmd ; Str.Delim delim ] -> + cmd ^ (String.make 1 (String.get delim 0)), (String.make 1 (String.get delim 1)) + | [ Str.Text cmd ; Str.Delim delim; Str.Text options ] -> + cmd ^ (String.make 1 (String.get delim 0)), (String.make 1 (String.get delim 1)) ^ options + | [ Str.Delim delim ] -> + (String.make 1 (String.get delim 0)), (String.make 1 (String.get delim 1)) + | [ Str.Delim delim; Str.Text options ] -> + (String.make 1 (String.get delim 0)), (String.make 1 (String.get delim 1)) ^ options + | _ -> assert false + +let opt_to_byte_options = + let regex_cmxs = Str.regexp ("\\([^/]+\\)[.]cmxs\\($\\|[ \t]\\)") in + fun options -> str_global_replace regex_cmxs "\\1.cmo\\2" options + let output_unix_error (exn : exn) = match exn with | Unix.Unix_error (error, _function, arg) -> @@ -228,20 +247,11 @@ let do_cmp = ref (if Sys.os_type="Win32" then !do_diffs else "cmp -s") let do_make = ref "make" let n = ref 4 (* the level of parallelism *) -let suites = ref [] -(** options appended to toplevel for all tests *) -let additional_options = ref "" -(** options prepended to toplevel for all tests *) -let additional_options_pre = ref "" (** special configuration, with associated oracles *) let special_config = ref "" let do_error_code = ref false -let exclude_suites = ref [] - -let exclude s = exclude_suites := s :: !exclude_suites - let xunit = ref false let io_mutex = Mutex.create () @@ -253,69 +263,25 @@ let lock_fprintf f = let lock_printf s = lock_fprintf Format.std_formatter s let lock_eprintf s = lock_fprintf Format.err_formatter s +let suites = ref [] let make_test_suite s = suites := s :: !suites -(* Those variables are read from a ptests_config file *) -let default_suites = ref [] -let toplevel_path = ref "" - -(* Splits the command string to separate the command name from the parameters - [let cmd_name,param=command_partition cmd in assert cmd=cmd_name^param] -*) -let command_partition = - let regexp_unescaped_blank = Str.regexp "[^\\ ] " in - fun cmd -> - match str_bounded_full_split regexp_unescaped_blank cmd 2 with - | [ Str.Text cmd ] -> - cmd, "" - | [ Str.Text cmd ; Str.Delim delim ] -> - cmd ^ (String.make 1 (String.get delim 0)), (String.make 1 (String.get delim 1)) - | [ Str.Text cmd ; Str.Delim delim; Str.Text options ] -> - cmd ^ (String.make 1 (String.get delim 0)), (String.make 1 (String.get delim 1)) ^ options - | [ Str.Delim delim ] -> - (String.make 1 (String.get delim 0)), (String.make 1 (String.get delim 1)) - | [ Str.Delim delim; Str.Text options ] -> - (String.make 1 (String.get delim 0)), (String.make 1 (String.get delim 1)) ^ options - | _ -> assert false +let exclude_suites = ref [] +let exclude s = exclude_suites := s :: !exclude_suites -let opt_to_byte_options = - let regex_cmxs = Str.regexp ("\\([^/]+\\)[.]cmxs\\($\\|[ \t]\\)") in - fun options -> str_global_replace regex_cmxs "\\1.cmo\\2" options +let macro_post_options = ref "" (* value set to @PTEST_POST_OPTIONS@ macro *) +let macro_pre_options = ref "" (* value set to @PTEST_PRE_OPTIONS@ macro *) +let macro_options = ref "@PTEST_PRE_OPTIONS@ @PTEST_OPT@ @PTEST_POST_OPTIONS@" +let macro_default_options = ref "-journal-disable -check -no-autoload-plugins" -let opt_to_byte cmd = - let opt_to_byte toplevel = - match string_del_suffix "frama-c" toplevel with - | Some path -> path ^ "frama-c.byte" - | None -> - match string_del_suffix "toplevel.opt" toplevel with - | Some path -> path ^ "toplevel.byte" - | None -> - match string_del_suffix "frama-c-gui" toplevel with - | Some path -> path ^ "frama-c-gui.byte" - | None -> - match string_del_suffix "viewer.opt" toplevel with - | Some path -> path ^ "viewer.byte" - | None -> toplevel - in - let cmdname, options = command_partition cmd in - (opt_to_byte cmdname) ^ (opt_to_byte_options options) - -let change_toplevel_to_gui () = - let s = !toplevel_path in - match string_del_suffix "toplevel.opt" s with - | Some s -> toplevel_path := s ^ "viewer.opt" - | None -> - match string_del_suffix "toplevel.byte" s with - | Some s -> toplevel_path := s ^ "viewer.byte" - | None -> - match string_del_suffix "frama-c" s with - | Some s -> toplevel_path := s ^ "frama-c-gui" - | None -> - match string_del_suffix "frama-c.byte" s with - | Some s -> toplevel_path := s ^ "frama-c-gui.byte" - | None -> () +let macro_frama_c_cmd = ref "@frama-c-exe@ @PTEST_DEFAULT_OPTIONS@" +let macro_frama_c = ref "@frama-c-exe@ @PTEST_DEFAULT_OPTIONS@ @PTEST_LOAD_OPTIONS@" +let default_toplevel = ref "@frama-c@" +(* Those variables are read from a ptests_config file *) +let toplevel_path = ref "" (* value set to @frama-c-exe@ macro *) +let default_suites = ref [] let () = Unix.putenv "LC_ALL" "C" (* some oracles, especially in Jessie, depend on the @@ -425,13 +391,13 @@ let rec argspec = " Use native toplevel (default)"; "-config", Arg.Set_string special_config, " <name> Use special configuration and oracles"; - "-add-options", Arg.Set_string additional_options, + "-add-options", Arg.Set_string macro_post_options, "<options> Add additional options to be passed to the toplevels \ that will be launched. <options> are added after standard test options"; - "-add-options-pre", Arg.Set_string additional_options_pre, + "-add-options-pre", Arg.Set_string macro_pre_options, "<options> Add additional options to be passed to the toplevels \ that will be launched. <options> are added before standard test options."; - "-add-options-post", Arg.Set_string additional_options, + "-add-options-post", Arg.Set_string macro_post_options, "Synonym of -add-options"; "-exclude", Arg.String exclude, "<name> Exclude a test or a suite from the run"; @@ -541,7 +507,35 @@ let () = end (** Must be done after reading config *) -let () = if !behavior = Gui then change_toplevel_to_gui () +let () = + if !use_byte then begin + match string_del_suffix "frama-c" !toplevel_path with + | Some path -> toplevel_path := path ^ "frama-c.byte" + | None -> + match string_del_suffix "toplevel.opt" !toplevel_path with + | Some path -> toplevel_path := path ^ "toplevel.byte" + | None -> + match string_del_suffix "frama-c-gui" !toplevel_path with + | Some path -> toplevel_path := path ^ "frama-c-gui.byte" + | None -> + match string_del_suffix "viewer.opt" !toplevel_path with + | Some path -> toplevel_path := path ^ "viewer.byte" + | None -> () + end; + if !behavior = Gui then begin + match string_del_suffix "toplevel.opt" !toplevel_path with + | Some s -> toplevel_path := s ^ "viewer.opt" + | None -> + match string_del_suffix "toplevel.byte" !toplevel_path with + | Some s -> toplevel_path := s ^ "viewer.byte" + | None -> + match string_del_suffix "frama-c" !toplevel_path with + | Some s -> toplevel_path := s ^ "frama-c-gui" + | None -> + match string_del_suffix "frama-c.byte" !toplevel_path with + | Some s -> toplevel_path := s ^ "frama-c-gui.byte" + | None -> () + end (* redefine name if special configuration expected *) let redefine_name name = @@ -594,6 +588,12 @@ end = struct end +type does_expand = { + has_ptest_file : bool; + has_ptest_opt : bool; + has_frama_c_exe : bool; +} + module Macros = struct module StringMap = Map.Make (String) @@ -618,6 +618,9 @@ struct let macro_regex = Str.regexp "@\\([-A-Za-z_0-9]+\\)@" in fun macros s -> let has_ptest_file = ref false in + let has_ptest_opt = ref false in + let has_ptest_options = ref false in + let has_frama_c_exe = ref false in if !verbosity >= 3 then lock_printf "%% Expand: %s@." s; if !verbosity >= 4 then print_macros macros; let rec aux s = @@ -627,9 +630,14 @@ struct if Str.string_match macro_regex s 0 then begin let macro = Str.matched_group 1 s in try + (match macro with + | "PTEST_FILE" -> has_ptest_file := true + | "PTEST_OPT" -> has_ptest_opt := true + | "PTEST_OPTIONS" -> has_ptest_options := true + | "frama-c-exe" -> has_frama_c_exe := true + | _ -> ()); if !verbosity >= 4 then lock_printf "%% - macro is %s\n%!" macro; let replacement = find macro macros in - if String.(macro = "PTEST_FILE") then has_ptest_file := true; if !verbosity >= 3 then lock_printf "%% - replacement for %s is %s\n%!" macro replacement; aux replacement @@ -645,7 +653,10 @@ struct let r = aux s in Mutex.unlock str_mutex; if !verbosity >= 3 then lock_printf "%% Expansion result: %s@." r; - !has_ptest_file, r + { has_ptest_file= !has_ptest_file; + has_ptest_opt= !has_ptest_opt; + has_frama_c_exe= !has_frama_c_exe; + }, r with e -> lock_eprintf "Uncaught exception %s\n%!" (Printexc.to_string e); Mutex.unlock str_mutex; @@ -670,7 +681,6 @@ struct add name (get name macros ^ expand macros def) macros end - type execnow = { ex_cmd: string; (** command to launch *) @@ -757,7 +767,13 @@ end = struct let default_macros () = let l = [ - "frama-c", !toplevel_path; + "frama-c-exe", !toplevel_path; + "frama-c-cmd", !macro_frama_c_cmd; + "frama-c", !macro_frama_c; + "PTEST_DEFAULT_OPTIONS", !macro_default_options; + "PTEST_OPTIONS", !macro_options; + "PTEST_PRE_OPTIONS", !macro_pre_options; + "PTEST_POST_OPTIONS", !macro_post_options; "PTEST_MAKE_MODULE", "make -s"; "PTEST_MODULE", ""; "PTEST_PLUGIN", ""; @@ -771,8 +787,8 @@ end = struct dc_execnow = []; dc_filter = None ; dc_exit_code = None; - dc_default_toplevel = !toplevel_path; - dc_commands = [ { toplevel= !toplevel_path; opts=default_options; macros=Macros.empty; exit_code=None; logs= []; timeout= ""} ]; + dc_default_toplevel = !default_toplevel; + dc_commands = [ { toplevel= !default_toplevel; opts=""; macros=Macros.empty; exit_code=None; logs= []; timeout= ""} ]; dc_dont_run = false; dc_load_module = ""; dc_cmxs_module = StringSet.empty; @@ -897,9 +913,10 @@ end = struct ("",current) deps in if String.(deps = "") then current - else + else begin let make_cmd = Macros.expand current.dc_macros "@PTEST_MAKE_MODULE@" in config_exec ~once:true ~file dir (make_cmd ^ deps) current + end let update_module_macros modules macros = let def = String.concat "," modules in @@ -1253,7 +1270,7 @@ end = struct "PTEST_DIR", SubDir.get cmd.directory; "PTEST_RESULT", SubDir.get cmd.directory ^ "/" ^ redefine_name "result"; - "PTEST_FILE", Filename.sanitize ptest_file; + "PTEST_FILE", ptest_file; "PTEST_NAME", ptest_name; "PTEST_NUMBER", string_of_int cmd.n; "PTEST_OPT", cmd.options; @@ -1265,8 +1282,29 @@ end = struct let macros = Macros.add_list macros cmd.macros in let macros = Macros.add_defaults ~defaults macros in let process_macros s = Macros.expand macros s in + let toplevel = + let in_toplevel,toplevel= Macros.does_expand macros cmd.toplevel in + if not cmd.execnow then begin + let has_ptest_file, options = + if in_toplevel.has_ptest_opt then in_toplevel.has_ptest_file, [] + else + let in_option,options= Macros.does_expand macros cmd.options in + (in_option.has_ptest_file || in_toplevel.has_ptest_file), + (if in_toplevel.has_frama_c_exe then + [ process_macros "@PTEST_PRE_OPTIONS@" ; + options ; + process_macros "@PTEST_POST_OPTIONS@" ; + ] + else [ options ]) + in + String.concat " " (toplevel::(if has_ptest_file then options else ptest_file::options)) + end + else toplevel + in { cmd with macros; + toplevel; + options = ""; (* no more usable *) log_files = List.map process_macros cmd.log_files; filter = match cmd.filter with @@ -1274,42 +1312,12 @@ end = struct | Some filter -> Some (process_macros filter) } - let contains_frama_c_binary_name = - Str.regexp "[^( ]*\\(toplevel\\|viewer\\|frama-c-gui\\|frama-c[^-]\\).*" - - let frama_c_binary_name = - Str.regexp "\\([^ ]*\\(toplevel\\|viewer\\|frama-c-gui\\|frama-c\\)\\(\\.opt\\|\\.byte\\|\\.exe\\)?\\)" - let basic_command_string = fun command -> - let macros = command.macros in - let has_ptest_file_t, toplevel = - Macros.does_expand macros command.toplevel - in - let has_ptest_file_o, options = Macros.does_expand macros command.options in - let toplevel = if !use_byte then opt_to_byte toplevel else toplevel in - let toplevel, contains_frama_c_binary = - str_string_match_and_replace contains_frama_c_binary_name - frama_c_binary_name ~suffix:" -check" toplevel - in - let options = - if contains_frama_c_binary - then begin - let opt_load = Macros.expand macros (Macros.get "PTEST_LOAD_OPTIONS" macros) in - let opt_pre = Macros.expand macros !additional_options_pre in - let opt_post = Macros.expand macros !additional_options in - String.concat " " [opt_load ; opt_pre ; options ; opt_post] - end else options - in - let options = if !use_byte then opt_to_byte_options options else options in let raw_command = - String.concat " " - (if has_ptest_file_t || has_ptest_file_o || command.execnow then - [ toplevel ; options] - else begin - let file = Filename.sanitize @@ get_ptest_file command in - [ toplevel ; file ; options] - end) + (* necessary until OPT are using direct -load-module option *) + if !use_byte then opt_to_byte_options command.toplevel + else command.toplevel in if command.timeout = "" then raw_command else "ulimit -t " ^ command.timeout ^ " && " ^ raw_command @@ -1553,12 +1561,7 @@ let do_command command = if !behavior <> Examine && not (!(execnow.ex_done) && execnow.ex_once) then begin remove_execnow_results execnow; - let cmd = - if !use_byte then - opt_to_byte execnow.ex_cmd - else - execnow.ex_cmd - in + let cmd = execnow.ex_cmd in if !verbosity >= 1 || !behavior = Show then begin lock_printf "%% launch %s@." cmd; end;