Skip to content

Commit

Permalink
Consistent exit strategy and prettier error messages
Browse files Browse the repository at this point in the history
  • Loading branch information
Stevendeo committed Oct 4, 2023
1 parent 9e5b4aa commit 8c951fe
Show file tree
Hide file tree
Showing 16 changed files with 239 additions and 91 deletions.
13 changes: 10 additions & 3 deletions src/bin/common/parse_command.ml
Original file line number Diff line number Diff line change
Expand Up @@ -324,7 +324,8 @@ let mk_execution_opt input_format parse_only ()
set_preludes preludes;
`Ok()

let mk_internal_opt disable_weaks enable_assertions warning_as_error gc_policy
let mk_internal_opt
disable_weaks enable_assertions warning_as_error continue_on_error gc_policy
=
let gc_policy = match gc_policy with
| 0 | 1 | 2 -> gc_policy
Expand All @@ -335,6 +336,7 @@ let mk_internal_opt disable_weaks enable_assertions warning_as_error gc_policy
set_disable_weaks disable_weaks;
set_enable_assertions enable_assertions;
set_warning_as_error warning_as_error;
set_exit_on_error (not continue_on_error);
`Ok(gc_policy)

let mk_limit_opt age_bound fm_cross_limit timelimit_interpretation
Expand Down Expand Up @@ -496,7 +498,7 @@ let halt_opt version_info where =
| Some w -> handle_where w; `Ok true
| None -> if version_info then (handle_version_info version_info; `Ok true)
else `Ok false
with Failure f -> `Error (false, f)
with Failure f -> `Error (false, f) (* TODO(Steven): dead code? *)
| Error (b, m) -> `Error (b, m)

let get_verbose_t =
Expand Down Expand Up @@ -779,6 +781,10 @@ let parse_internal_opt =
let doc = "Enable warning as error" in
Arg.(value & flag & info ["warning-as-error"] ~docs ~doc) in

let continue_on_error =
let doc = "Sets Alt-ergo's behavior to continue on errors" in
Arg.(value & flag & info ["continue-on-error"] ~docs ~doc) in

let gc_policy =
let doc =
"Set the gc policy allocation. 0 = next-fit policy, 1 = \
Expand All @@ -788,7 +794,8 @@ let parse_internal_opt =
Arg.(value & opt int 0 & info ["gc-policy"] ~docv ~docs ~doc) in

Term.(ret (const mk_internal_opt $
disable_weaks $ enable_assertions $ warning_as_error $ gc_policy
disable_weaks $ enable_assertions $
warning_as_error $ continue_on_error $ gc_policy
))

let parse_limit_opt =
Expand Down
128 changes: 81 additions & 47 deletions src/bin/common/solving_loop.ml
Original file line number Diff line number Diff line change
Expand Up @@ -50,8 +50,34 @@ let empty_solver_ctx = {
global = [];
}

let unsupported_opt () =
Printer.print_std "unsupported"
let recoverable_error ?(code = 1) =
Format.kasprintf (fun msg ->
let () =
if msg <> "" then
match Options.get_output_format () with
| Smtlib2 -> Printer.print_smtlib_err "%s" msg
| _ -> Printer.print_err "%s" msg
in
if Options.get_exit_on_error () then exit code)

let fatal_error ?(code = 1) =
Format.kasprintf (fun msg -> recoverable_error ~code "%s" msg; exit code)

let exit_as_timeout () = fatal_error ~code:142 "timeout"

let warning (msg : ('a, Format.formatter, unit, unit, unit, 'b) format6) : 'a =
if Options.get_warning_as_error () then
recoverable_error msg
else
Printer.print_wrn msg

let unsupported_opt opt =
let () =
match Options.get_output_format () with
| Options.Smtlib2 -> Printer.print_std "unsupported"
| _ -> ()
in
warning "unsupported option %s" opt

let main () =
let () = Dolmen_loop.Code.init [] in
Expand Down Expand Up @@ -104,7 +130,7 @@ let main () =
Some partial_model
| `Unsat -> None
with Util.Timeout ->
if not (Options.get_timelimit_per_goal()) then exit 142;
if not (Options.get_timelimit_per_goal()) then exit_as_timeout ();
None
in

Expand Down Expand Up @@ -182,14 +208,18 @@ let main () =
with
| Util.Timeout ->
FE.print_status (FE.Timeout None) 0;
exit 142
exit_as_timeout ()
| Parsing.Parse_error ->
Printer.print_err "%a" Errors.report
(Syntax_error ((Lexing.dummy_pos,Lexing.dummy_pos),""));
exit 1
(* TODO(Steven): display a dummy value is a bad idea.
This should only be executed with the legacy frontend, which should
be deprecated in a near future, so this code will be removed (or at
least, its behavior unspecified). *)
fatal_error
"%a"
Errors.report
(Syntax_error ((Lexing.dummy_pos,Lexing.dummy_pos),""))
| Errors.Error e ->
Printer.print_err "%a" Errors.report e;
exit 1
fatal_error "%a" Errors.report e
in

let all_used_context = FE.init_all_used_context () in
Expand All @@ -203,9 +233,13 @@ let main () =
List.fold_left (typed_loop all_used_context) { state with env; } l
with
| Errors.Error e ->
if e != Warning_as_error then
Printer.print_err "%a" Errors.report e;
exit 1
let () =
if e != Warning_as_error then
recoverable_error "%a" Errors.report e
else
recoverable_error ""
in
state
end
in

Expand All @@ -219,7 +253,7 @@ let main () =
Options.Time.unset_timeout ();
with Util.Timeout ->
FE.print_status (FE.Timeout None) 0;
exit 142
exit_as_timeout ()
in

let solver_ctx_key: solver_ctx State.key =
Expand Down Expand Up @@ -250,21 +284,18 @@ let main () =
else
st, `Continue stmt
in
let handle_exn _ bt = function
let handle_exn st bt = function
| Dolmen.Std.Loc.Syntax_error (_, `Regular msg) ->
Printer.print_err "%t" msg;
exit 1
recoverable_error "%t" msg; st
| Util.Timeout ->
Printer.print_status_timeout None None None None;
exit 142
| Errors.Error e ->
Printer.print_err "%a" Errors.report e;
exit 1
exit_as_timeout ()
| Errors.Error e -> recoverable_error "%a" Errors.report e; st
| _ as exn -> Printexc.raise_with_backtrace exn bt
in
let finally ~handle_exn st e =
match e with
| Some (bt, exn) -> handle_exn st bt exn; st
| Some (bt, exn) -> handle_exn st bt exn
| _ -> st
in
let set_output_format fmt =
Expand All @@ -273,8 +304,9 @@ let main () =
| ".ae" -> Options.set_output_format Native
| ".smt2" | ".psmt2" -> Options.set_output_format Smtlib2
| s ->
Printer.print_wrn
"The output format %s is not supported by the Dolmen frontend." s
warning
"The output format %s is not supported by the Dolmen frontend."
s
in
(* The function In_channel.input_all is not available before OCaml 4.14. *)
let read_all ch =
Expand Down Expand Up @@ -348,7 +380,8 @@ let main () =
in

let print_wrn_opt ~name loc ty value =
Printer.print_wrn "%a The option %s expects a %s, got %a"
warning
"%a The option %s expects a %s, got %a"
Loc.report loc name ty DStd.Term.print value
in

Expand All @@ -369,9 +402,11 @@ let main () =
solver Tableaux. *)
if Stdlib.(Options.get_sat_solver () = Tableaux) then
Options.set_unsat_core true
else Printer.print_wrn "%a The generation of unsat cores is not \
supported for the current SAT solver. Please \
choose the SAT solver Tableaux."
else
warning
"%a The generation of unsat cores is not \
supported for the current SAT solver. Please \
choose the SAT solver Tableaux."
Loc.report st_loc
| ":produce-unsat-cores", Symbol { name = Simple "false"; _ } ->
Options.set_unsat_core false
Expand Down Expand Up @@ -403,11 +438,9 @@ let main () =
| ":print-success"
| ":random-seed"), _
->
unsupported_opt ();
Printer.print_wrn "unsupported option '%s'" name
unsupported_opt name
| _ ->
unsupported_opt ();
Printer.print_err "unknown option '%s'" name
unsupported_opt name
in

let handle_get_info (st : State.t) (name: string) =
Expand All @@ -417,8 +450,7 @@ let main () =
in
let pp_reason_unknown st =
let err () =
let msg = "Invalid (get-info :reason-unknown)" in
Printer.print_smtlib_err "%s" msg
recoverable_error "Invalid (get-info :reason-unknown)"
in
match State.get partial_model_key st with
| None -> err ()
Expand All @@ -434,7 +466,13 @@ let main () =
| ":authors" ->
print_std (fun fmt -> Fmt.pf fmt "%S") "Alt-Ergo developers"
| ":error-behavior" ->
print_std Fmt.string "immediate-exit"
let behavior =
if Options.get_exit_on_error () then
"immediate-exit"
else
"continued-execution"
in
print_std Fmt.string behavior
| ":name" ->
print_std (fun fmt -> Fmt.pf fmt "%S") "Alt-Ergo"
| ":reason-unknown" ->
Expand All @@ -443,11 +481,9 @@ let main () =
print_std Fmt.string Version._version
| ":all-statistics"
| ":assertion-stack-levels" ->
unsupported_opt ();
Printer.print_wrn "unsupported option '%s'" name
unsupported_opt name
| _ ->
unsupported_opt ();
Printer.print_err "unknown option '%s'" name
unsupported_opt name
in

let handle_stmt :
Expand Down Expand Up @@ -541,15 +577,13 @@ let main () =
end
| None ->
(* TODO: add the location of the statement. *)
Printer.print_smtlib_err "No model produced.";
st
recoverable_error "No model produced."; st
else
begin
(* TODO: add the location of the statement. *)
Printer.print_smtlib_err
"Model generation disabled (try --produce-models)";
st
end
(* TODO: add the location of the statement. *)
let () =
recoverable_error "Model generation disabled (try --produce-models)"
in
st

| {contents = `Get_info kind; _ } ->
handle_get_info st kind;
Expand Down Expand Up @@ -611,7 +645,7 @@ let main () =
State.flush st () |> ignore
with exn ->
let bt = Printexc.get_raw_backtrace () in
handle_exn st bt exn
ignore (handle_exn st bt exn)
in

let filename = get_file () in
Expand Down
16 changes: 6 additions & 10 deletions src/lib/structures/errors.ml
Original file line number Diff line number Diff line change
Expand Up @@ -240,24 +240,20 @@ let report_run_error fmt = function

let report fmt = function
| Parser_error s ->
Options.pp_comment fmt
(Format.sprintf "Parser Error: %s" s);
Format.fprintf fmt "Parser Error: %s" s;
| Lexical_error (l,s) ->
Loc.report fmt l;
Options.pp_comment fmt
(Format.sprintf "Lexical Error: %s" s);
Format.fprintf fmt "Lexical Error: %s" s;
| Syntax_error (l,s) ->
Loc.report fmt l;
Options.pp_comment fmt
(Format.sprintf "Syntax Error: %s" s);
Format.fprintf fmt "Syntax Error: %s" s;
| Typing_error (l,e) ->
Loc.report fmt l;
Options.pp_comment fmt "Typing Error: ";
Format.fprintf fmt "Typing Error: ";
report_typing_error fmt e
| Run_error e ->
Options.pp_comment fmt "Fatal Error: ";
Format.fprintf fmt "Fatal Error: ";
report_run_error fmt e
| Dolmen_error (code, descr) ->
Options.pp_comment fmt
(Format.sprintf "Error %s (code %i)" descr code);
Format.fprintf fmt "Error %s (code %i)" descr code;
| Warning_as_error -> ()
3 changes: 3 additions & 0 deletions src/lib/util/options.ml
Original file line number Diff line number Diff line change
Expand Up @@ -302,14 +302,17 @@ let get_type_smt2 () = !type_smt2
let disable_weaks = ref false
let enable_assertions = ref false
let warning_as_error = ref false
let exit_on_error = ref true

let set_disable_weaks b = disable_weaks := b
let set_enable_assertions b = enable_assertions := b
let set_warning_as_error b = warning_as_error := b
let set_exit_on_error b = exit_on_error := b

let get_disable_weaks () = !disable_weaks
let get_enable_assertions () = !enable_assertions
let get_warning_as_error () = !warning_as_error
let get_exit_on_error () = !exit_on_error

(** Limit options *)

Expand Down
7 changes: 7 additions & 0 deletions src/lib/util/options.mli
Original file line number Diff line number Diff line change
Expand Up @@ -359,6 +359,9 @@ val set_enable_assertions : bool -> unit
(** Set [warning_as_error] accessible with {!val:get_warning_as_error} *)
val set_warning_as_error : bool -> unit

(** Set [exit_on_error] accessible with {!val:get_exit_on_error} *)
val set_exit_on_error : bool -> unit

(** Set [timelimit_interpretation] accessible with
{!val:get_timelimit_interpretation} *)
val set_timelimit_interpretation : float -> unit
Expand Down Expand Up @@ -658,6 +661,10 @@ val get_enable_assertions : unit -> bool
val get_warning_as_error : unit -> bool
(** Default to [false] *)

(** [true] if alt-ergo exits on all errors. *)
val get_exit_on_error : unit -> bool
(** Default to [true] *)

(** {4 Limit options} *)

(** Value specifying the age limit bound. *)
Expand Down
8 changes: 4 additions & 4 deletions tests/cram.t/run.t
Original file line number Diff line number Diff line change
Expand Up @@ -50,21 +50,21 @@ combinations of produce-models et al.
First, if model generation is not enabled, we should error out when a
`(get-model)` statement is issued:

$ echo '(get-model)' | alt-ergo --frontend dolmen -i smtlib2 -o smtlib2
$ echo '(get-model)' | alt-ergo --frontend dolmen -i smtlib2 -o smtlib2 --continue-on-error
(error "Model generation disabled (try --produce-models)")

This should be the case Tableaux solver as well:

$ echo '(get-model)' | alt-ergo --frontend dolmen --sat-solver Tableaux -i smtlib2 -o smtlib2
$ echo '(get-model)' | alt-ergo --frontend dolmen --sat-solver Tableaux -i smtlib2 -o smtlib2 --continue-on-error
(error "Model generation disabled (try --produce-models)")

The messages above mention `--produce-models`, but we can also use
`set-option`.

$ echo '(get-model)' | alt-ergo --frontend dolmen --produce-models -i smtlib2 -o smtlib2
$ echo '(get-model)' | alt-ergo --frontend dolmen --produce-models -i smtlib2 -o smtlib2 --continue-on-error
(error "No model produced.")

$ echo '(set-option :produce-models true)(get-model)' | alt-ergo --frontend dolmen --sat-solver Tableaux -i smtlib2 -o smtlib2
$ echo '(set-option :produce-models true)(get-model)' | alt-ergo --frontend dolmen --sat-solver Tableaux -i smtlib2 -o smtlib2 --continue-on-error
(error "No model produced.")

And now some cases where it should work (using either `--produce-models` or `set-option`):
Expand Down
Loading

0 comments on commit 8c951fe

Please sign in to comment.