Skip to content
Snippets Groups Projects
GuiProver.ml 6.43 KiB
(**************************************************************************)
(*                                                                        *)
(*  This file is part of WP plug-in of Frama-C.                           *)
(*                                                                        *)
(*  Copyright (C) 2007-2019                                               *)
(*    CEA (Commissariat a l'energie atomique et aux energies              *)
(*         alternatives)                                                  *)
(*                                                                        *)
(*  you can redistribute it and/or modify it under the terms of the GNU   *)
(*  Lesser General Public License as published by the Free Software       *)
(*  Foundation, version 2.1.                                              *)
(*                                                                        *)
(*  It is distributed in the hope that it will be useful,                 *)
(*  but WITHOUT ANY WARRANTY; without even the implied warranty of        *)
(*  MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the         *)
(*  GNU Lesser General Public License for more details.                   *)
(*                                                                        *)
(*  See the GNU Lesser General Public License version 2.1                 *)
(*  for more details (enclosed in the file licenses/LGPLv2.1).            *)
(*                                                                        *)
(**************************************************************************)

let no_status = `Share "theme/default/never_tried.png"
let ok_status = `Share "theme/default/surely_valid.png"
let ko_status = `Share "theme/default/unknown.png"
let wg_status = `Share "theme/default/invalid.png"

let filter = function
  | VCS.Qed | VCS.Tactical | VCS.NativeCoq -> false
  | VCS.Why3 _ | VCS.NativeAltErgo -> true

(* -------------------------------------------------------------------------- *)
(* --- Palette Tool                                                       --- *)
(* -------------------------------------------------------------------------- *)

let timeout_for = function
  | VCS.NativeAltErgo | VCS.Why3 _ ->
      let value = Wp_parameters.Timeout.get () in
      let spin = new Widget.spinner
        ~tooltip:"Prover Timeout (0 for none)"
        ~min:0 ~step:5 ~value () in
      Some spin
  | _ -> None

let stepout_for = function
  | VCS.NativeAltErgo ->
      let value = Wp_parameters.Steps.get () in
      let spin = new Widget.spinner
        ~tooltip:"Prover Step Limit (0 for none)"
        ~min:0 ~step:100 ~value () in
      Some spin
  | _ -> None

class prover ~(console:Wtext.text) ~prover =
  let tooltip = "Configure Prover" in
  let content = new Wpane.form () in
  let result = new Widget.label ~style:`Code ~align:`Center ~text:"No Result" () in
  let timeout = timeout_for prover in
  let stepout = stepout_for prover in
  object(self)
    inherit Wpalette.tool ~tooltip ~content:content#widget ()
    initializer
      begin
        assert (filter prover) ;
        content#add_row ~xpadding:6 ~ypadding:4 result#coerce ;
        Wutil.on timeout (fun spin -> content#add_field ~label:"Timeout" spin#coerce) ;
        Wutil.on stepout (fun spin -> content#add_field ~label:"Steps" spin#coerce) ;
      end

    method prover = prover

    method private log wpo res =
      begin
        let fout = Wpo.get_file_logout wpo prover in
        let ferr = Wpo.get_file_logerr wpo prover in
        let lout = Sys.file_exists fout in
        let lerr = Sys.file_exists ferr in
        if lout || lerr then console#hrule ;
        console#scroll () ;
        console#printf "[%a] %a@." VCS.pp_prover prover VCS.pp_result res ;
        if lout then Command.pp_from_file console#fmt fout ;
        if lerr then Command.pp_from_file console#fmt ferr ;
        if lout || lerr then console#hrule ;
      end

    method private run wpo =
      begin
        let spinner = function None -> None | Some s -> Some s#get in
        let config = {
          VCS.valid = false ;
          VCS.timeout = spinner timeout ;
          VCS.stepout = spinner stepout ;
        } in
        let result wpo _prv _res = self#update wpo in
        let task = Prover.prove ~config ~result wpo prover in
        let thread = Task.thread task in
        let kill () =
          Wpo.set_result wpo prover VCS.no_result ;
          Task.cancel thread in
        Wpo.set_result wpo prover (VCS.computing kill) ;
        let server = ProverTask.server () in
        Task.spawn server thread ;
        Task.launch server ;
        Wutil.later (fun () -> self#update wpo) ;
      end

    method clear =
      begin
        self#set_status no_status ;
        self#set_action ~icon:`MEDIA_PLAY ~tooltip:"Run Prover" ?callback:None () ;
        Pretty_utils.ksfprintf self#set_label "%a" VCS.pp_prover prover ;
        result#set_text "No Goal" ;
      end

    method update wpo =
      begin
        let res = Wpo.get_result wpo prover in
        result#set_text (Pretty_utils.to_string VCS.pp_result_perf res) ;
        match res.VCS.verdict with
        | VCS.NoResult ->
            let callback () = self#run wpo in
            self#set_status no_status ;
            self#set_action ~icon:`MEDIA_PLAY ~tooltip:"Run Prover" ~callback () ;
        | VCS.Computing callback ->
            self#set_status `EXECUTE ;
            self#set_action ~tooltip:"Interrrupt Prover" ~icon:`STOP ~callback () ;
            Pretty_utils.ksfprintf self#set_label "%a (...)" VCS.pp_prover prover ;
        | VCS.Valid | VCS.Checked ->
            let callback () = self#run wpo in
            self#set_status ok_status ;
            self#set_action ~tooltip:"Run Prover" ~icon:`MEDIA_PLAY ~callback () ;
            Pretty_utils.ksfprintf self#set_label "%a (%a)" VCS.pp_prover prover
              Rformat.pp_time res.VCS.prover_time ;
        | VCS.Invalid | VCS.Unknown | VCS.Timeout | VCS.Stepout ->
            let callback () = self#run wpo in
            self#set_status ko_status ;
            self#set_action ~tooltip:"Run Prover" ~icon:`MEDIA_PLAY ~callback () ;
            Pretty_utils.ksfprintf self#set_label "%a (?)" VCS.pp_prover prover ;
        | VCS.Failed ->
            let callback () = self#log wpo res in
            self#set_status `DIALOG_WARNING ;
            self#set_action ~tooltip:"Dump Logs" ~icon:`FILE ~callback () ;
            Pretty_utils.ksfprintf self#set_label "%a (failed)" VCS.pp_prover prover ;
      end

  end