Skip to content

Commit

Permalink
Handling error strategy (#853)
Browse files Browse the repository at this point in the history
Fixes #851
  • Loading branch information
Stevendeo authored Oct 9, 2023
1 parent c54fe39 commit 99755e7
Show file tree
Hide file tree
Showing 17 changed files with 212 additions and 93 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 @@ -319,7 +319,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 @@ -330,6 +331,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 @@ -491,7 +493,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 @@ -774,6 +776,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 @@ -783,7 +789,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
126 changes: 83 additions & 43 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 @@ -103,9 +129,8 @@ let main () =
| `Sat partial_model | `Unknown partial_model ->
Some partial_model
| `Unsat -> None
with
| Util.Timeout ->
if not (Options.get_timelimit_per_goal()) then exit 142;
with Util.Timeout ->
if not (Options.get_timelimit_per_goal()) then exit_as_timeout ();
None
in

Expand Down Expand Up @@ -186,14 +211,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): displaying 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 @@ -207,9 +236,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
| Exit -> exit 0
end
in
Expand All @@ -224,7 +257,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 @@ -255,22 +288,21 @@ 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
exit_as_timeout ()
| Errors.Error e ->
Printer.print_err "%a" Errors.report e;
exit 1
recoverable_error "%a" Errors.report e;
st
| Exit -> exit 0
| _ 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 @@ -279,8 +311,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 @@ -353,7 +386,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 @@ -374,9 +408,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 @@ -408,11 +444,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 @@ -422,8 +456,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 @@ -439,7 +472,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 @@ -448,11 +487,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 @@ -530,6 +567,10 @@ let main () =
handle_option loc name value;
st

| {contents = `Set_option _; _} ->
recoverable_error "Invalid set-option";
st

| {contents = `Get_model; _ } ->
if Options.get_interpretation () then
match State.get partial_model_key st with
Expand All @@ -546,12 +587,11 @@ 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
recoverable_error
"Model generation disabled (try --produce-models)";
st
end
Expand Down Expand Up @@ -623,7 +663,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 @@ -299,14 +299,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 @@ -356,6 +356,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 @@ -653,6 +656,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
1 change: 1 addition & 0 deletions src/lib/util/printer.ml
Original file line number Diff line number Diff line change
Expand Up @@ -386,6 +386,7 @@ let print_status_preprocess ?(validity_mode=true)

let print_smtlib_err ?(flushed=true) s =
(* The smtlib error messages are printed on the regular output. *)
pp_std_smt ();
let fmt = Options.Output.get_fmt_regular () in
let k fmt =
if flushed || Options.get_output_with_forced_flush () then
Expand Down
Loading

0 comments on commit 99755e7

Please sign in to comment.