Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Print unknown reason in SMT-LIB syntax #1

Closed
Closed
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
1 change: 1 addition & 0 deletions src/bin/common/parse_command.ml
Original file line number Diff line number Diff line change
Expand Up @@ -362,6 +362,7 @@ let mk_limit_opt age_bound fm_cross_limit timelimit_interpretation
set_fm_cross_limit fm_cross_limit;
set_timelimit_interpretation timelimit_interpretation;
set_steps_bound steps_bound;
Steps.set_steps_bound steps_bound;
set_timelimit timelimit;
set_timelimit_per_goal timelimit_per_goal;
`Ok()
Expand Down
18 changes: 17 additions & 1 deletion src/bin/common/solving_loop.ml
Original file line number Diff line number Diff line change
Expand Up @@ -182,7 +182,7 @@ let main () =
let ur = SAT.get_unknown_reason partial_model in
Printer.print_fmt (Options.Output.get_fmt_diagnostic ())
"@[<v 0>Returned unknown reason = %a@]"
Sat_solver_sig.pp_unknown_reason_opt ur;
Sat_solver_sig.pp_ae_unknown_reason_opt ur;
FE.print_model (Options.Output.get_fmt_models ()) partial_model
end;
Some mdl
Expand Down Expand Up @@ -338,6 +338,15 @@ let main () =
State.create_key ~pipe:"" "named_terms"
in

let set_steps_bound i st =
try DO.Steps.set i st with
Invalid_argument _ -> (* Raised by Steps.set_steps_bound *)
fatal_error
"Error while setting steps bound to %i: current step = %i."
i
(Steps.get_steps ())
in

let debug_parsed_pipe st c =
if State.get State.debug st then
Format.eprintf "[logic][parsed][%a] @[<hov>%a@]@."
Expand Down Expand Up @@ -496,6 +505,7 @@ let main () =
sat;
st
| Tableaux | Tableaux_CDCL | CDCL | CDCL_Tableaux ->
O.set_sat_solver sat;
(* `make_sat` returns the sat solver corresponding to the new sat_solver
option. *)
DO.SatSolver.set sat st
Expand Down Expand Up @@ -627,6 +637,12 @@ let main () =
| Some b ->
set_optimize b st
end
| ":steps-bound", Symbol { name = Simple level; _ } ->
begin
match int_of_string_opt level with
| None -> print_wrn_opt ~name st_loc "integer" value; st
| Some i -> set_steps_bound i st
end
| _ ->
unsupported_opt name; st
in
Expand Down
9 changes: 2 additions & 7 deletions src/bin/js/worker_js.ml
Original file line number Diff line number Diff line change
Expand Up @@ -179,13 +179,8 @@ let main worker_id content =
raise Exit
| Errors.Error e ->
begin match e with
| Errors.Run_error r -> begin
match r with
| Steps_limit _ ->
returned_status :=
Worker_interface.LimitReached "Steps limit"
| _ -> returned_status := Worker_interface.Error "Run error"
end
| Errors.Run_error _ ->
returned_status := Worker_interface.Error "Run error"
| _ -> returned_status := Worker_interface.Error "Error"
end;
Printer.print_err "%a" Errors.report e;
Expand Down
6 changes: 6 additions & 0 deletions src/lib/frontend/d_state_option.ml
Original file line number Diff line number Diff line change
Expand Up @@ -99,6 +99,12 @@ let msatsolver =

module SatSolver = (val msatsolver)

let msteps =
let on_update _ sat st = Steps.set_steps_bound sat; st in
(create_opt ~on_update "steps_bound" O.get_steps_bound)

module Steps = (val msteps)

(* Some options can be initialized to gain some performance. *)
let options_requiring_initialization = [
(module SatSolverModule : S);
Expand Down
6 changes: 6 additions & 0 deletions src/lib/frontend/d_state_option.mli
Original file line number Diff line number Diff line change
Expand Up @@ -64,5 +64,11 @@ module SatSolver : S with type t = Util.sat_solver
on SatSolver: when SatSolver is updated, this one also is. *)
module SatSolverModule : Accessor with type t = (module Sat_solver_sig.S)

(** Option for setting the max number of steps. Interfaces with the toplevel
Steps module.
The [set] function may raise Invalid_arg from the Steps.set_steps_bound call
if the new option value is lower than the current number of steps. *)
module Steps : S with type t = int

(** Initializes the state with options that requires some preprocessing. *)
val init : D_loop.Typer.state -> D_loop.Typer.state
91 changes: 60 additions & 31 deletions src/lib/frontend/frontend.ml
Original file line number Diff line number Diff line change
Expand Up @@ -142,7 +142,7 @@ module type S = sig
expl : Explanation.t
}

type 'a process = ?loc : Loc.t -> env -> 'a -> env
type 'a process = ?loc : Loc.t -> 'a -> env -> env

val init_env : ?sat_env:sat_env -> used_context -> env

Expand Down Expand Up @@ -226,7 +226,7 @@ module Make(SAT : Sat_solver_sig.S) : S with type sat_env = SAT.t = struct
expl : Explanation.t
}

type 'a process = ?loc : Loc.t -> env -> 'a -> env
type 'a process = ?loc : Loc.t -> 'a -> env -> env

let init_env ?(sat_env=SAT.empty ()) used_context =
{
Expand Down Expand Up @@ -298,13 +298,16 @@ module Make(SAT : Sat_solver_sig.S) : S with type sat_env = SAT.t = struct
if Options.get_unsat_core () then Ex.singleton (Ex.RootDep {name;f;loc})
else Ex.empty

let internal_push ?(loc = Loc.dummy) (env : env) (n : int) : env =
let internal_push ?(loc = Loc.dummy) (n : int) (env : env) : env =
ignore loc;
Util.loop ~f:(fun _ res () -> Stack.push res env.consistent_dep_stack)
~max:n ~elt:(env.res, env.expl) ~init:();
{env with sat_env = SAT.push env.sat_env n}
let sat_env =
Steps.apply_without_step_limit (fun () -> SAT.push env.sat_env n)
in
{env with sat_env}

let internal_pop ?(loc = Loc.dummy) (env : env) (n : int) : env =
let internal_pop ?(loc = Loc.dummy) (n : int) (env : env) : env =
ignore loc;
let res, expl =
Util.loop ~f:(fun _n () _env -> Stack.pop env.consistent_dep_stack)
Expand All @@ -315,8 +318,8 @@ module Make(SAT : Sat_solver_sig.S) : S with type sat_env = SAT.t = struct

let internal_assume
?(loc = Loc.dummy)
(env : env)
((name, f, mf) : string * E.t * bool) =
((name, f, mf) : string * E.t * bool)
(env : env) =
let is_hyp = try (Char.equal '@' name.[0]) with _ -> false in
if not is_hyp && unused_context name env.used_context then
env
Expand Down Expand Up @@ -349,15 +352,15 @@ module Make(SAT : Sat_solver_sig.S) : S with type sat_env = SAT.t = struct
in
{env with sat_env; expl}

let internal_pred_def ?(loc = Loc.dummy) env (name, f) =
let internal_pred_def ?(loc = Loc.dummy) (name, f) env =
if unused_context name env.used_context then
env
else
let expl = mk_root_dep name f loc in
let sat_env = SAT.pred_def env.sat_env f name expl loc in
{env with sat_env; expl}

let internal_query ?(loc = Loc.dummy) env (n, f, sort) =
let internal_query ?(loc = Loc.dummy) (n, f, sort) env =
ignore loc;
let expl =
match env.res with
Expand Down Expand Up @@ -386,8 +389,8 @@ module Make(SAT : Sat_solver_sig.S) : S with type sat_env = SAT.t = struct

let internal_th_assume
?(loc = Loc.dummy)
env
({ Expr.ax_name; Expr.ax_form ; _ } as th_elt) =
({ Expr.ax_name; Expr.ax_form ; _ } as th_elt)
env =
if unused_context ax_name env.used_context then
env
else
Expand All @@ -398,43 +401,69 @@ module Make(SAT : Sat_solver_sig.S) : S with type sat_env = SAT.t = struct
{env with sat_env; expl}
| `Unsat -> env

let handle_sat_exn f ?loc env x =
try f ?loc env x with
let check_step_limit f env =
match SAT.get_unknown_reason env.sat_env with
| Some (Step_limit _) -> env
| _ -> f env

let handle_sat_exn f ?loc x env =
try f ?loc x env with
| SAT.Sat t -> {env with res = `Sat t}
| SAT.Unsat expl -> {env with res = `Unsat; expl = Ex.union expl env.expl}
| SAT.I_dont_know t -> {env with res = `Unknown t}
(* The SAT.Timeout exception is not catched. *)

let push = handle_sat_exn internal_push
(* Wraps the function f to check if the step limit is reached (in which case,
don't do anything), and then calls the function & catches the
exceptions. *)
let wrap_f f ?loc x env =
check_step_limit (handle_sat_exn f ?loc x) env

let push = wrap_f internal_push

let pop = handle_sat_exn internal_pop
let pop = wrap_f internal_pop

let assume = handle_sat_exn internal_assume
let assume = wrap_f internal_assume

let pred_def = handle_sat_exn internal_pred_def
let pred_def = wrap_f internal_pred_def

let query = handle_sat_exn internal_query
let query = wrap_f internal_query

let th_assume = handle_sat_exn internal_th_assume
let th_assume = wrap_f internal_th_assume

let process_decl ?(hook_on_status=(fun _ -> ignore)) env d =
try
match d.st_decl with
| Push n -> internal_push ~loc:d.st_loc env n
| Pop n -> internal_pop ~loc:d.st_loc env n
| Assume (n, f, mf) -> internal_assume ~loc:d.st_loc env (n, f, mf)
| PredDef (f, name) -> internal_pred_def ~loc:d.st_loc env (name, f)
| Push n -> check_step_limit (internal_push ~loc:d.st_loc n) env
| Pop n -> check_step_limit (internal_pop ~loc:d.st_loc n) env
| Assume (n, f, mf) ->
check_step_limit (internal_assume ~loc:d.st_loc (n, f, mf)) env
| PredDef (f, name) ->
check_step_limit (internal_pred_def ~loc:d.st_loc (name, f)) env
| RwtDef _ -> assert false
| Query (n, f, sort) ->
begin
let env = internal_query ~loc:d.st_loc env (n, f, sort) in
match env.res with
| `Unsat ->
hook_on_status (Unsat (d, env.expl)) (Steps.get_steps ());
env
| _ -> assert false
(* If we have reached an unknown state, we can return it right
away. *)
match SAT.get_unknown_reason env.sat_env with
| Some (Step_limit _) -> raise (SAT.I_dont_know env.sat_env)
| Some _ ->
(* For now, only the step limit is an unknown step reachable
here. We could raise SAT.I_dont_know as in the previous case,
but we have choosen a defensive strategy. *)
assert false
| None ->
let env =
internal_query ~loc:d.st_loc (n, f, sort) env
in
match env.res with
| `Unsat ->
hook_on_status (Unsat (d, env.expl)) (Steps.get_steps ());
env
| _ -> assert false
end
| ThAssume th_elt -> internal_th_assume ~loc:d.st_loc env th_elt
| ThAssume th_elt ->
check_step_limit (internal_th_assume ~loc:d.st_loc th_elt) env
with
| SAT.Sat t ->
(* This case should mainly occur when a query has a non-unsat result,
Expand Down Expand Up @@ -479,7 +508,7 @@ module Make(SAT : Sat_solver_sig.S) : S with type sat_env = SAT.t = struct
far. You may need to change your model generation strategy \
or to increase your timeouts. \
Returned unknown reason = %a@]"
Sat_solver_sig.pp_unknown_reason_opt ur;
Sat_solver_sig.pp_ae_unknown_reason_opt ur;

| Some (lazy model) ->
Models.output_concrete_model ppf model
Expand Down
2 changes: 1 addition & 1 deletion src/lib/frontend/frontend.mli
Original file line number Diff line number Diff line change
Expand Up @@ -68,7 +68,7 @@ module type S = sig
They catch the [Sat], [Unsat] and [I_dont_know] exceptions to update the
frontend environment, but not the [Timeout] exception which is raised to
the user. *)
type 'a process = ?loc:Loc.t -> env -> 'a -> env
type 'a process = ?loc:Loc.t -> 'a -> env -> env

val push : int process

Expand Down
8 changes: 7 additions & 1 deletion src/lib/reasoners/fun_sat.ml
Original file line number Diff line number Diff line change
Expand Up @@ -1630,7 +1630,8 @@ module Make (Th : Theory.S) : Sat_solver_sig.S = struct
{ acc.guards with
current_guard = new_guard;
guards = guards;
}})
}}
)
~max:to_push
~elt:()
~init:env
Expand Down Expand Up @@ -1735,6 +1736,7 @@ module Make (Th : Theory.S) : Sat_solver_sig.S = struct
assert (Ex.has_no_bj dep);
dep
| Util.Timeout -> model_gen_on_timeout env
| Util.Step_limit_reached n -> i_dont_know env (Step_limit n)

let add_guard env gf =
let current_guard = env.guards.current_guard in
Expand All @@ -1753,6 +1755,10 @@ module Make (Th : Theory.S) : Sat_solver_sig.S = struct
(* don't attempt to compute a model if timeout before
calling unsat function *)
i_dont_know env (Timeout Assume)
| Util.Step_limit_reached n ->
(* When reaching the step limit on an assume, we do not want to answer
'unknown' right away. *)
{env with unknown_reason = Some (Step_limit n)}

let pred_def env f name dep _loc =
Debug.pred_def f;
Expand Down
26 changes: 20 additions & 6 deletions src/lib/reasoners/sat_solver_sig.ml
Original file line number Diff line number Diff line change
Expand Up @@ -35,21 +35,35 @@ type timeout_reason =
| Assume
| ProofSearch
| ModelGen
[@@deriving show]

let pp_smt_timeout_reason ppf = function
| Assume -> Fmt.pf ppf ":assume"
| ProofSearch -> Fmt.pf ppf ":proof-search"
| ModelGen -> Fmt.pf ppf ":model-gen"

let pp_ae_timeout_reason ppf = function
| Assume -> Fmt.pf ppf "Assume"
| ProofSearch -> Fmt.pf ppf "ProofSearch"
| ModelGen -> Fmt.pf ppf "ModelGen"

type unknown_reason =
| Incomplete
| Memout
| Step_limit of int
| Timeout of timeout_reason

let pp_unknown_reason ppf = function
| Incomplete -> Fmt.pf ppf "Incomplete"
| Memout -> Fmt.pf ppf "Memout"
| Timeout t -> Fmt.pf ppf "Timeout:%a" pp_timeout_reason t
| Incomplete -> Fmt.pf ppf "incomplete"
| Memout -> Fmt.pf ppf "memout"
| Step_limit i -> Fmt.pf ppf "(:step-limit %i)" i
| Timeout t -> Fmt.pf ppf "(:timeout %a)" pp_smt_timeout_reason t

let pp_unknown_reason_opt ppf = function
let pp_ae_unknown_reason_opt ppf = function
| None -> Fmt.pf ppf "Decided"
| Some ur -> pp_unknown_reason ppf ur
| Some Incomplete -> Fmt.pf ppf "Incomplete"
| Some Memout -> Fmt.pf ppf "Memout"
| Some Step_limit i -> Fmt.pf ppf "StepLimit:%i" i
| Some Timeout t -> Fmt.pf ppf "Timeout:%a" pp_ae_timeout_reason t

module type S = sig
type t
Expand Down
6 changes: 5 additions & 1 deletion src/lib/reasoners/sat_solver_sig.mli
Original file line number Diff line number Diff line change
Expand Up @@ -36,10 +36,14 @@ type timeout_reason =
type unknown_reason =
| Incomplete
| Memout
| Step_limit of int
| Timeout of timeout_reason

(** Prints the unknown reason in the default SMT-LIB format *)
val pp_unknown_reason: unknown_reason Fmt.t
val pp_unknown_reason_opt : unknown_reason option Fmt.t

(** Prints an optional unknown reason in Alt-Ergo format *)
val pp_ae_unknown_reason_opt : unknown_reason option Fmt.t

module type S = sig
type t
Expand Down
Loading