From 702c1731552b4e00dd0ca928f03c1383ed0dc98d Mon Sep 17 00:00:00 2001 From: Alasdair Date: Mon, 26 Feb 2024 15:58:37 +0000 Subject: [PATCH] Add an option to choose counterexample solver --- .github/workflows/coverage.yml | 3 ++- src/sail_smt_backend/sail_plugin_smt.ml | 10 +++++++++- src/sail_smt_backend/smtlib.ml | 22 +++++++++++++++++----- 3 files changed, 28 insertions(+), 7 deletions(-) diff --git a/.github/workflows/coverage.yml b/.github/workflows/coverage.yml index 638ee4b0c..c3275f1de 100644 --- a/.github/workflows/coverage.yml +++ b/.github/workflows/coverage.yml @@ -20,7 +20,8 @@ jobs: - name: System dependencies run: | - sudo apt install build-essential libgmp-dev z3 cvc4 opam cargo + sudo apt-get update + sudo apt-get -o Acquire::Retries=3 install build-essential libgmp-dev z3 cvc4 opam cargo - name: Restore cached opam id: cache-opam-restore diff --git a/src/sail_smt_backend/sail_plugin_smt.ml b/src/sail_smt_backend/sail_plugin_smt.ml index 079de23b1..472972d39 100644 --- a/src/sail_smt_backend/sail_plugin_smt.ml +++ b/src/sail_smt_backend/sail_plugin_smt.ml @@ -69,9 +69,17 @@ open Libsail let opt_includes_smt : string list ref = ref [] +let set_auto_solver arg = + let open Smtlib in + match counterexample_solver_from_name arg with Some solver -> opt_auto_solver := solver | None -> () + let smt_options = [ - ("-smt_auto", Arg.Tuple [Arg.Set Jib_smt.opt_auto], " generate SMT and automatically call the CVC5 solver"); + ("-smt_auto", Arg.Tuple [Arg.Set Jib_smt.opt_auto], " generate SMT and automatically call the smt solver"); + ( "-smt_auto_solver", + Arg.String set_auto_solver, + " set the solver to use for counterexample checks (default cvc5)" + ); ("-smt_ignore_overflow", Arg.Set Jib_smt.opt_ignore_overflow, " ignore integer overflow in generated SMT"); ("-smt_propagate_vars", Arg.Set Jib_smt.opt_propagate_vars, " propgate variables through generated SMT"); ( "-smt_int_size", diff --git a/src/sail_smt_backend/smtlib.ml b/src/sail_smt_backend/smtlib.ml index bac8d5543..279cf4765 100644 --- a/src/sail_smt_backend/smtlib.ml +++ b/src/sail_smt_backend/smtlib.ml @@ -70,6 +70,15 @@ open Libsail open Ast open Ast_util +type counterexample_solver = Cvc5 | Cvc4 | Z3 + +let counterexample_command = function Cvc5 -> "cvc5 --lang=smt2.6" | Cvc4 -> "cvc4 --lang=smt2.6" | Z3 -> "z3" + +let counterexample_solver_from_name name = + match String.lowercase_ascii name with "cvc4" -> Some Cvc4 | "cvc5" -> Some Cvc5 | "z3" -> Some Z3 | _ -> None + +let opt_auto_solver = ref Cvc5 + type smt_typ = | Bitvec of int | Bool @@ -655,7 +664,7 @@ let rec run frame = let check_counterexample ast env fname function_id args arg_ctyps arg_smt_names = let open Printf in prerr_endline ("Checking counterexample: " ^ fname); - let in_chan = ksprintf Unix.open_process_in "cvc5 --lang=smt2.6 %s" fname in + let in_chan = ksprintf Unix.open_process_in "%s %s" (counterexample_command !opt_auto_solver) fname in let lines = ref [] in begin try @@ -667,10 +676,10 @@ let check_counterexample ast env fname function_id args arg_ctyps arg_smt_names let solver_output = List.rev !lines |> String.concat "\n" |> parse_sexps in begin match solver_output with - | Atom "sat" :: List model :: _ -> + | Atom "sat" :: (List (Atom "model" :: model) | List model) :: _ -> let open Value in let open Interpreter in - prerr_endline (sprintf "Solver found counterexample: %s" Util.("ok" |> green |> clear)); + print_endline (sprintf "Solver found counterexample: %s" Util.("ok" |> green |> clear)); let counterexample = build_counterexample args arg_ctyps arg_smt_names model in List.iter (fun (id, v) -> prerr_endline (" " ^ string_of_id id ^ " -> " ^ string_of_value v)) counterexample; let istate = initial_state ast env !primops in @@ -690,10 +699,13 @@ let check_counterexample ast env fname function_id args arg_ctyps arg_smt_names begin match result with | Some (V_bool false) | None -> - ksprintf prerr_endline "Replaying counterexample: %s" Util.("ok" |> green |> clear) + ksprintf print_endline "Replaying counterexample: %s" Util.("ok" |> green |> clear) | _ -> () end - | _ -> prerr_endline "Solver could not find counterexample" + | _ -> + print_endline "Solver could not find counterexample"; + print_endline "Solver output:"; + List.iter print_endline !lines end; let status = Unix.close_process_in in_chan in ()