Skip to content

Commit

Permalink
add a --solver CLI option :-)
Browse files Browse the repository at this point in the history
  • Loading branch information
zapashcanon committed Jun 7, 2024
1 parent 9876974 commit b5588c3
Show file tree
Hide file tree
Showing 15 changed files with 67 additions and 35 deletions.
3 changes: 2 additions & 1 deletion CHANGES.md
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
## unreleased

- support the extended const proposal
- support other solvers through the `--solver` option (Z3, Colibri2, Bitwuzla and CVC5)
- support the extended constant expressions proposal
- `owi opt` and `owi sym` can run on `.wasm` files directly
- remove dependency on `wabt`
- better API for `Parse`, `Compile` and `Simplified` (renamed to `Binary`), added a `Binary_to_text` module
Expand Down
33 changes: 23 additions & 10 deletions src/bin/owi.ml
Original file line number Diff line number Diff line change
Expand Up @@ -22,6 +22,18 @@ let dir_file =
let parse s = `Ok (Fpath.v s) in
(parse, Fpath.pp)

let solver_conv =
Cmdliner.Arg.conv
( Smtml.Solver_dispatcher.solver_type_of_string
, Smtml.Solver_dispatcher.pp_solver_type )

let deterministic_result_order =
let doc =
"Guarantee a fixed deterministic order of found failures. This implies \
--no-stop-at-failure."
in
Cmdliner.Arg.(value & flag & info [ "deterministic-result-order" ] ~doc)

let files =
let doc = "source files" in
let f = existing_non_dir_file in
Expand All @@ -39,13 +51,6 @@ let no_values =
let doc = "do not display a value for each symbol" in
Cmdliner.Arg.(value & flag & info [ "no-value" ] ~doc)

let deterministic_result_order =
let doc =
"Guarantee a fixed deterministic order of found failures. This implies \
--no-stop-at-failure."
in
Cmdliner.Arg.(value & flag & info [ "deterministic-result-order" ] ~doc)

let optimize =
let doc = "optimize mode" in
Cmdliner.Arg.(value & flag & info [ "optimize" ] ~doc)
Expand All @@ -54,6 +59,13 @@ let profiling =
let doc = "profiling mode" in
Cmdliner.Arg.(value & flag & info [ "profiling"; "p" ] ~doc)

let solver =
let doc = "SMT solver to use" in
Cmdliner.Arg.(
value
& opt solver_conv Smtml.Solver_dispatcher.Z3_solver
& info [ "solver"; "s" ] ~doc )

let unsafe =
let doc = "skip typechecking pass" in
Cmdliner.Arg.(value & flag & info [ "unsafe"; "u" ] ~doc)
Expand Down Expand Up @@ -123,7 +135,8 @@ let c_cmd =
Term.(
const Cmd_c.cmd $ debug $ arch $ property $ testcomp $ output $ workers
$ opt_lvl $ includes $ files $ profiling $ unsafe $ optimize
$ no_stop_at_failure $ no_values $ deterministic_result_order $ concolic )
$ no_stop_at_failure $ no_values $ deterministic_result_order $ concolic
$ solver )

let fmt_cmd =
let open Cmdliner in
Expand Down Expand Up @@ -189,7 +202,7 @@ let sym_cmd =
Term.(
const Cmd_sym.cmd $ profiling $ debug $ unsafe $ optimize $ workers
$ no_stop_at_failure $ no_values $ deterministic_result_order $ workspace
$ files )
$ solver $ files )

let conc_cmd =
let open Cmdliner in
Expand All @@ -202,7 +215,7 @@ let conc_cmd =
Term.(
const Cmd_conc.cmd $ profiling $ debug $ unsafe $ optimize $ workers
$ no_stop_at_failure $ no_values $ deterministic_result_order $ workspace
$ files )
$ solver $ files )

let wasm2wat_cmd =
let open Cmdliner in
Expand Down
4 changes: 2 additions & 2 deletions src/cmd/cmd_c.ml
Original file line number Diff line number Diff line change
Expand Up @@ -95,7 +95,7 @@ let metadata ~workspace arch property files : unit Result.t =

let cmd debug arch property _testcomp workspace workers opt_lvl includes files
profiling unsafe optimize no_stop_at_failure no_values
deterministic_result_order concolic : unit Result.t =
deterministic_result_order concolic solver : unit Result.t =
if debug then Logs.set_level (Some Debug);
let workspace = Fpath.v workspace in
let includes = C_share.lib_location @ includes in
Expand All @@ -106,4 +106,4 @@ let cmd debug arch property _testcomp workspace workers opt_lvl includes files
let files = [ modul ] in
(if concolic then Cmd_conc.cmd else Cmd_sym.cmd)
profiling debug unsafe optimize workers no_stop_at_failure no_values
deterministic_result_order workspace files
deterministic_result_order workspace solver files
1 change: 1 addition & 0 deletions src/cmd/cmd_c.mli
Original file line number Diff line number Diff line change
Expand Up @@ -19,4 +19,5 @@ val cmd :
-> bool
-> bool
-> bool
-> Smtml.Solver_dispatcher.solver_type
-> unit Result.t
4 changes: 2 additions & 2 deletions src/cmd/cmd_conc.ml
Original file line number Diff line number Diff line change
Expand Up @@ -590,7 +590,7 @@ let launch solver tree link_state modules_to_run =
which are handled here. Most of the computations are done in the Result
monad, hence the let*. *)
let cmd profiling debug unsafe optimize workers no_stop_at_failure no_values
deterministic_result_order (workspace : Fpath.t) files =
deterministic_result_order (workspace : Fpath.t) solver files =
ignore (workers, no_stop_at_failure, deterministic_result_order, workspace);

if profiling then Log.profiling_on := true;
Expand All @@ -599,7 +599,7 @@ let cmd profiling debug unsafe optimize workers no_stop_at_failure no_values
(* deterministic_result_order implies no_stop_at_failure *)
(* let no_stop_at_failure = deterministic_result_order || no_stop_at_failure in *)
let* _created_dir = Bos.OS.Dir.create ~path:true ~mode:0o755 workspace in
let solver = Solver.fresh () in
let solver = Solver.fresh solver () in
let* link_state, modules_to_run =
simplify_then_link_files ~unsafe ~optimize files
in
Expand Down
1 change: 1 addition & 0 deletions src/cmd/cmd_conc.mli
Original file line number Diff line number Diff line change
Expand Up @@ -12,5 +12,6 @@ val cmd :
-> bool
-> bool
-> Fpath.t
-> Smtml.Solver_dispatcher.solver_type
-> Fpath.t list
-> unit Result.t
4 changes: 2 additions & 2 deletions src/cmd/cmd_sym.ml
Original file line number Diff line number Diff line change
Expand Up @@ -235,7 +235,7 @@ let run_file ~unsafe ~optimize pc filename =
which are handled here. Most of the computations are done in the Result
monad, hence the let*. *)
let cmd profiling debug unsafe optimize workers no_stop_at_failure no_values
deterministic_result_order (workspace : Fpath.t) files =
deterministic_result_order (workspace : Fpath.t) solver files =
if profiling then Log.profiling_on := true;
if debug then Log.debug_on := true;
(* deterministic_result_order implies no_stop_at_failure *)
Expand All @@ -244,7 +244,7 @@ let cmd profiling debug unsafe optimize workers no_stop_at_failure no_values
let pc = Choice.return (Ok ()) in
let result = List.fold_left (run_file ~unsafe ~optimize) pc files in
let thread : Thread.t = Thread.create () in
let results = Symbolic_choice.run ~workers result thread in
let results = Symbolic_choice.run ~workers solver result thread in
let print_bug = function
| `ETrap (tr, model) ->
Format.pp_std "Trap: %s@\n" (Trap.to_string tr);
Expand Down
1 change: 1 addition & 0 deletions src/cmd/cmd_sym.mli
Original file line number Diff line number Diff line change
Expand Up @@ -12,5 +12,6 @@ val cmd :
-> bool
-> bool
-> Fpath.t
-> Smtml.Solver_dispatcher.solver_type
-> Fpath.t list
-> unit Result.t
6 changes: 4 additions & 2 deletions src/symbolic/solver.ml
Original file line number Diff line number Diff line change
Expand Up @@ -6,8 +6,10 @@ type 'a solver_module = (module Smtml.Solver_intf.S with type t = 'a)

type t = S : ('a solver_module * 'a) -> t [@@unboxed]

let fresh () =
let module Mapping = Smtml.Z3_mappings.Fresh.Make () in
let fresh solver () =
let module Mapping = (val Smtml.Solver_dispatcher.mappings_of_solver solver)
in
let module Mapping = Mapping.Fresh.Make () in
let module Batch = Smtml.Solver.Batch (Mapping) in
let solver = Batch.create ~logic:QF_BVFP () in
S ((module Batch), solver)
Expand Down
2 changes: 1 addition & 1 deletion src/symbolic/solver.mli
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@

type t

val fresh : unit -> t
val fresh : Smtml.Solver_dispatcher.solver_type -> unit -> t

val check : t -> Smtml.Expr.t list -> Smtml.Solver_intf.satisfiability

Expand Down
23 changes: 13 additions & 10 deletions src/symbolic/symbolic_choice.ml
Original file line number Diff line number Diff line change
Expand Up @@ -65,7 +65,12 @@ module CoreImpl : sig

type 'a run_result = ('a eval * Thread.t) Seq.t

val run : workers:int -> 'a t -> Thread.t -> 'a run_result
val run :
workers:int
-> Smtml.Solver_dispatcher.solver_type
-> 'a t
-> Thread.t
-> 'a run_result
end = struct
module Schedulable = struct
(*
Expand Down Expand Up @@ -162,15 +167,13 @@ end = struct
let spawn_worker sched wls_init =
Wq.make_pledge sched.res_writer;
Domain.spawn (fun () ->
let wls = wls_init () in
Fun.protect
~finally:(fun () -> Wq.end_pledge sched.res_writer)
~finally:(fun () ->
Wq.end_pledge sched.res_writer;
Wq.fail sched.work_queue )
(fun () ->
try work wls sched
with e ->
let bt = Printexc.get_raw_backtrace () in
Wq.fail sched.work_queue;
Printexc.raise_with_backtrace e bt ) )
let wls = wls_init () in
work wls sched ) )
end

module State = struct
Expand Down Expand Up @@ -295,13 +298,13 @@ end = struct

type 'a run_result = ('a eval * Thread.t) Seq.t

let run ~workers t thread =
let run ~workers solver t thread =
let open Scheduler in
let sched = init_scheduler () in
add_init_task sched (State.run t thread);
let join_handles =
Array.map
(fun () -> spawn_worker sched Solver.fresh)
(fun () -> spawn_worker sched (Solver.fresh solver))
(Array.init workers (Fun.const ()))
in
Wq.read_as_seq sched.res_writer ~finalizer:(fun () ->
Expand Down
7 changes: 6 additions & 1 deletion src/symbolic/symbolic_choice.mli
Original file line number Diff line number Diff line change
Expand Up @@ -13,4 +13,9 @@ include
and type 'a run_result = ('a eval * Thread.t) Seq.t
and module V := Symbolic_value

val run : workers:int -> 'a t -> Thread.t -> 'a run_result
val run :
workers:int
-> Smtml.Solver_dispatcher.solver_type
-> 'a t
-> Thread.t
-> 'a run_result
2 changes: 1 addition & 1 deletion src/symbolic/symbolic_choice_minimalist.ml
Original file line number Diff line number Diff line change
Expand Up @@ -63,4 +63,4 @@ let solver = M (fun st sol -> (Ok sol, st))

let add_pc (_vb : vbool) = return ()

let run ~workers:_ t thread = run t thread (Solver.fresh ())
let run ~workers:_ solver t thread = run t thread (Solver.fresh solver ())
7 changes: 6 additions & 1 deletion src/symbolic/symbolic_choice_minimalist.mli
Original file line number Diff line number Diff line change
Expand Up @@ -12,4 +12,9 @@ include
and type 'a run_result = ('a, err) Stdlib.Result.t * Thread.t
and module V := Symbolic_value

val run : workers:int -> 'a t -> Thread.t -> 'a run_result
val run :
workers:int
-> Smtml.Solver_dispatcher.solver_type
-> 'a t
-> Thread.t
-> 'a run_result
4 changes: 2 additions & 2 deletions test/fuzz/interprets.ml
Original file line number Diff line number Diff line change
Expand Up @@ -84,8 +84,8 @@ module Owi_symbolic : INTERPRET = struct
let c = Interpret.SymbolicM.modul link_state.envs regular in
let init_thread : Thread.t = Thread.create () in
let res, _ =
Symbolic_choice_minimalist.run ~workers:dummy_workers_count c
init_thread
Symbolic_choice_minimalist.run ~workers:dummy_workers_count
Smtml.Solver_dispatcher.Z3_solver c init_thread
in
match res with
| Ok res -> res
Expand Down

0 comments on commit b5588c3

Please sign in to comment.