why3session_latex.ml 14.3 KB
Newer Older
Andrei Paskevich's avatar
Andrei Paskevich committed
1 2 3
(********************************************************************)
(*                                                                  *)
(*  The Why3 Verification Platform   /   The Why3 Development Team  *)
4
(*  Copyright 2010-2015   --   INRIA - CNRS - Paris-Sud University  *)
Andrei Paskevich's avatar
Andrei Paskevich committed
5 6 7 8
(*                                                                  *)
(*  This software is distributed under the terms of the GNU Lesser  *)
(*  General Public License version 2.1, with the special exception  *)
(*  on linking described in file LICENSE.                           *)
9
(*                                                                  *)
Andrei Paskevich's avatar
Andrei Paskevich committed
10
(********************************************************************)
11 12 13 14 15

open Why3
open Why3session_lib
open Format

16
module Hprover = Whyconf.Hprover
17 18
module S = Session

19 20
let opt_style = ref 1
let opt_output_dir = ref ""
21
let opt_longtable = ref false
22 23 24 25 26 27 28 29
let opt_elements = ref None

let add_element s =
  let l = match !opt_elements with
    | None -> [s]
    | Some r -> s :: r
  in
  opt_elements := Some l
30 31

let spec =
32
  ("-style",
33
   Arg.Set_int opt_style,
34
   "<n> set output style (1 or 2, default 1)") ::
35
  ("-o",
36
   Arg.Set_string opt_output_dir,
37
   "<dir> where to produce LaTeX files (default: session dir)") ::
Andrei Paskevich's avatar
Andrei Paskevich committed
38
  ("-e",
39 40
   Arg.String add_element,
   "<path> produce a table for the element denoted by <path>") ::
41 42
  ("-longtable",
   Arg.Set opt_longtable,
MARCHE Claude's avatar
MARCHE Claude committed
43
   " use 'longtable' environment instead of 'tabular'") ::
44
  common_options
45 46 47 48 49 50 51 52 53 54 55 56 57 58


(* Statistics in LaTeX*)

let rec transf_depth tr =
  List.fold_left (fun depth g -> max depth (goal_depth g)) 0 tr.S.transf_goals
and goal_depth g =
  S.PHstr.fold
    (fun _st tr depth -> max depth (1 + transf_depth tr))
    g.S.goal_transformations 0

let theory_depth t =
  List.fold_left (fun depth g -> max depth (goal_depth g)) 0 t.S.theory_goals

59 60 61 62
let file_depth f =
  List.fold_left (fun depth t -> max depth (theory_depth t)) 0
    f.S.file_theories

63 64
let provers_latex_stats_goal provers goal =
  S.goal_iter_proof_attempt (fun a ->
65
    Hprover.replace provers a.S.proof_prover a.S.proof_prover) goal
66 67

let provers_latex_stats provers theory =
68
  S.theory_iter_proof_attempt (fun a ->
69
    Hprover.replace provers a.S.proof_prover a.S.proof_prover) theory
70

71 72
let provers_latex_stats_file provers file =
  S.file_iter_proof_attempt (fun a ->
73
    Hprover.replace provers a.S.proof_prover a.S.proof_prover) file
74

75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93
let protect s =
  let b = Buffer.create 7 in
  for i = 0 to String.length s - 1 do
    match s.[i] with
	'_' -> Buffer.add_string b "\\_"
      | c -> Buffer.add_char b c
  done;
  Buffer.contents b


let column n depth provers =
  if n == 1 then
    if depth > 1 then
      (List.length provers) + depth
    else
      (List.length provers) + depth + 1
  else
    (List.length provers) +  1

94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110
let print_head n depth provers fmt =
  if n == 1 then
    if (depth > 1) then
      fprintf fmt "\\hline \\multicolumn{%d}{|c|}{Proof obligations } "
        depth
    else
      fprintf fmt "\\hline \\multicolumn{%d}{|c|}{Proof obligations } "
        (depth + 1)
  else
    fprintf fmt "\\hline Proof obligations ";
  List.iter (fun a -> fprintf fmt "& \\provername{%a} " Whyconf.print_prover a)
    provers;
  fprintf fmt "\\\\ @."

let print_tabular_head n depth provers fmt =
  fprintf fmt "\\begin{tabular}";
  fprintf fmt "{| l |";
111
  for _i = 0 to (List.length provers) + depth do fprintf fmt "c |" done;
112 113 114 115 116 117
  fprintf fmt "}@.";
  print_head n depth provers fmt

let print_tabular_foot fmt =
  fprintf fmt "\\hline \\end{tabular}@."

118 119 120 121 122 123 124

let print_result_prov proofs prov fmt=
  List.iter (fun p ->
  try
    let pr = S.PHprover.find proofs p in
    let s = pr.S.proof_state in
      match s with
125
	| Session.Done res ->
126 127
	    begin
	      match res.Call_provers.pr_answer with
Claude Marche's avatar
Claude Marche committed
128 129 130 131 132 133 134 135
		| Call_provers.Valid ->
                  fprintf fmt "& \\valid{%.2f} " res.Call_provers.pr_time
		| Call_provers.Invalid ->
                  fprintf fmt "& \\invalid{%.2f} " res.Call_provers.pr_time
		| Call_provers.Timeout ->
                  fprintf fmt "& \\timeout{%ds} " pr.S.proof_timelimit
		| Call_provers.OutOfMemory ->
                  fprintf fmt "& \\outofmemory{%dM} " pr.S.proof_memlimit
136 137
		| Call_provers.StepLimitExceeded ->
                  fprintf fmt "& \\steplimitexceeded "
Claude Marche's avatar
Claude Marche committed
138 139 140 141 142 143
		| Call_provers.Unknown _ ->
                  fprintf fmt "& \\unknown{%.2f} " res.Call_provers.pr_time
		| Call_provers.Failure _ ->
                  fprintf fmt "& \\failure "
		| Call_provers.HighFailure ->
                  fprintf fmt "& \\highfailure "
Andrei Paskevich's avatar
Andrei Paskevich committed
144

145
	    end
146
	| Session.InternalFailure _ -> fprintf fmt "& Internal Failure"
147 148 149 150
	| Session.Interrupted -> fprintf fmt "& Not yet run"
	| Session.Unedited -> fprintf fmt "& Not yet edited"
	| Session.Scheduled | Session.Running
	| Session.JustEdited -> assert false
151 152 153 154 155 156 157 158 159
  with Not_found -> fprintf fmt "& \\noresult") prov;
  fprintf fmt "\\\\ @."


let rec goal_latex_stat fmt prov depth depth_max subgoal g =
  let column = column 1 depth prov
  in
    if depth > 0 then
      if subgoal > 0 then
160
	fprintf fmt "\\cline{%d-%d}@." (depth + 1) column
161 162
      else ()
    else
163
      fprintf fmt "\\hline@.";
164 165 166 167 168
    if depth_max > 1 then
      begin
	if subgoal > 0 then
	  begin
	    if(depth < depth_max)  then
169
	      for _i = 1 to depth do
170
                fprintf fmt " & "
171 172
              done
	    else
173
	      for _i = 1 to depth - 1 do
174
                fprintf fmt " & "
175 176 177 178 179
              done
	  end
	else
	  if(depth < depth_max) then
	    if depth > 0 then
180
              fprintf fmt " & "
181 182 183 184
      end
    else
      begin
	if subgoal > 0  then
185
	  for _i = 1 to depth do fprintf fmt " & " done
186
	else
187
	  if depth > 0 then fprintf fmt " & "
188 189 190 191
      end;
    if (depth <= 1) then
      fprintf fmt "\\explanation{%s} " (protect (S.goal_expl g))
    else
192
      fprintf fmt " " ;
193 194 195 196 197 198
    let proofs = g.S.goal_external_proofs in
      if (S.PHprover.length proofs) > 0 then
	begin
	  if depth_max <= 1 then
	    begin
	      if depth > 0 then
199
		for _i = depth to (depth_max - depth) do
200
                  fprintf fmt "& " done
201
	      else
202
		for _i = depth to (depth_max - depth - 1) do
203
                  fprintf fmt "& " done
204 205 206
	    end
	  else
	    if depth > 0 then
207
	      for _i = depth to (depth_max - depth - 1) do
208
		fprintf fmt "& " done
209
	    else
210
	      for _i = depth to (depth_max - depth - 2) do
211
		fprintf fmt "& " done;
212 213 214 215 216
	  print_result_prov proofs prov fmt;
	end;
      let tr = g.S.goal_transformations in
	if S.PHstr.length tr > 0 then
	  if S.PHprover.length proofs > 0 then
217
	    fprintf fmt "\\cline{%d-%d}@." (depth + 2) column;
218 219 220 221 222 223 224 225
	S.PHstr.iter (fun _st tr ->
          let goals = tr.S.transf_goals in
	  let _ = List.fold_left (fun subgoal g ->
              goal_latex_stat fmt prov (depth + 1) depth_max (subgoal) g;
               subgoal + 1) 0 goals in
	    () ) tr


226
let style_2_row fmt ?(transf=false) depth prov subgoal expl=
227
  let column = column 2 depth prov in
228
  if depth > 0 || transf then
229
    fprintf fmt "\\cline{%d-%d}@." 2 column
230
  else
231
    fprintf fmt "\\hline@.";
232
  for _i = 1 to depth do fprintf fmt "\\quad" done;
233 234 235
  let macro = if transf then "transformation" else "explanation" in
  if depth = 0 || transf then
    fprintf fmt "\\%s{%s} " macro expl
236
  else
237
    fprintf fmt "\\subgoal{%s}{%d} " expl (subgoal + 1)
Claude Marche's avatar
Claude Marche committed
238 239

let rec goal_latex2_stat fmt prov depth depth_max subgoal g =
240
  let proofs = g.S.goal_external_proofs in
241 242 243 244 245 246
  if S.PHprover.length proofs > 0 then
    begin
      style_2_row fmt depth prov subgoal (protect (S.goal_expl g));
      print_result_prov proofs prov fmt
    end
 else
247
    if (*depth = 0*) true then
248 249 250 251 252
      begin
        style_2_row fmt depth prov subgoal (protect (S.goal_expl g));
	fprintf fmt "& \\multicolumn{%d}{|c|}{}\\\\ @."
          (List.length prov)
      end;
253 254 255 256
  let tr = g.S.goal_transformations in
  if S.PHstr.length tr > 0 then
    begin
      S.PHstr.iter (fun _st tr ->
Andrei Paskevich's avatar
Andrei Paskevich committed
257
        style_2_row fmt ~transf:true (depth+1) prov subgoal
258
          (protect tr.S.transf_name);
259
	fprintf fmt "& \\multicolumn{%d}{|c|}{}\\\\@."
Claude Marche's avatar
Claude Marche committed
260
          (List.length prov);
261 262 263 264 265 266
	let goals = tr.S.transf_goals in
	let _ = List.fold_left (fun subgoal g ->
	  goal_latex2_stat fmt prov (depth + 1) depth_max (subgoal) g;
	  subgoal + 1) 0 goals in
	() ) tr
    end
267 268
 else
    if (S.PHprover.length proofs) == 0 && depth <> 0 then
269
      fprintf fmt "\\\\@."
270

271 272
let latex_tabular_goal n fmt depth provers g =
  print_tabular_head n depth provers fmt;
273
  if n == 1 then
274
    goal_latex_stat fmt provers 0 depth 0 g
275
  else
276 277
    goal_latex2_stat fmt provers 0 depth  0 g;
  print_tabular_foot fmt
278

279
let latex_tabular_aux n fmt depth provers t =
280 281 282
  if n == 1 then
    List.iter (goal_latex_stat fmt provers 0 depth 0) t.S.theory_goals
  else
283 284 285 286 287 288 289 290 291 292 293
    List.iter (goal_latex2_stat fmt provers 0 depth  0) t.S.theory_goals

let latex_tabular n fmt depth provers t =
  print_tabular_head n depth provers fmt;
  latex_tabular_aux n fmt depth provers t;
  print_tabular_foot fmt


let latex_tabular_file n fmt depth provers f =
  print_tabular_head n depth provers fmt;
  List.iter (latex_tabular_aux n fmt depth provers) f.S.file_theories;
294
  print_tabular_foot fmt
295 296 297 298 299


let latex_longtable n fmt depth name provers t=
  fprintf fmt "\\begin{longtable}";
  fprintf fmt "{| l |";
300
  for _i = 0 to (List.length provers) + depth do fprintf fmt "c |" done;
301
  fprintf fmt "}@.";
302
  (* First head *)
303 304
  print_head n depth provers fmt;
  fprintf fmt "\\hline \\endfirsthead @.";
305
  (* Other heads : "Continued... " added *)
306 307 308 309 310
  fprintf fmt "\\multicolumn{%d}{r}{\\textit{Continued from previous page}} \
\\\\ @." (List.length provers + 1) ;
  fprintf fmt "\\hline @.";
  print_head n depth provers fmt;
  fprintf fmt "\\hline \\endhead @.";
311
  (* Other foots : "Continued..." added *)
312 313
  fprintf fmt "\\hline \\multicolumn{%d}{r}{\\textit{Continued on next page}} \
\\\\ @." (List.length provers);
314
  (* Last foot : nothing *)
315 316 317 318 319 320 321 322 323 324
  fprintf fmt "\\endfoot \\endlastfoot @.";
  if n == 1 then
    List.iter (goal_latex_stat fmt provers 0 depth 0) t.S.theory_goals
  else
    List.iter (goal_latex2_stat fmt provers 0 depth  0) t.S.theory_goals;
  fprintf fmt "\\hline \\caption{%s statistics} @." (protect name);
  fprintf fmt "\\label{%s} \\end{longtable}@." (protect name)


let theory_latex_stat n table dir t =
325
  let provers = Hprover.create 9 in
326
  provers_latex_stats provers t;
327
  let provers = Hprover.fold (fun _ pr acc -> pr :: acc)
328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344
    provers [] in
  let provers = List.sort Whyconf.Prover.compare provers in
  let depth = theory_depth  t in
  let name = t.S.theory_name.Ident.id_string in
  let ch = open_out (Filename.concat dir(name^".tex")) in
  let fmt = formatter_of_out_channel ch in
    if table = "tabular" then
      begin
      latex_tabular n fmt depth provers t
      end
    else
      latex_longtable n fmt depth name provers t;
    close_out ch

let file_latex_stat n table  dir f =
   List.iter (theory_latex_stat n table dir) f.S.file_theories

345
let standalone_goal_latex_stat n _table dir g =
346
  let provers = Hprover.create 9 in
347
  provers_latex_stats_goal provers g;
348
  let provers = Hprover.fold (fun _ pr acc -> pr :: acc)
349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365
    provers [] in
  let provers = List.sort Whyconf.Prover.compare provers in
  let depth = goal_depth g in
  let name = g.S.goal_name.Ident.id_string in
  let ch = open_out (Filename.concat dir(name^".tex")) in
  let fmt = formatter_of_out_channel ch in
  latex_tabular_goal n fmt depth provers g;
(*
  if table = "tabular" then
  begin
  latex_tabular_goal n fmt depth provers g
  end
  else
  latex_longtable_goal n fmt depth name provers g;
*)
close_out ch

366
let file_latex_stat_all n _table dir f =
367
  let provers = Hprover.create 9 in
368
  provers_latex_stats_file provers f;
369
  let provers = Hprover.fold (fun _ pr acc -> pr :: acc)
370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389
    provers [] in
  let provers = List.sort Whyconf.Prover.compare provers in
  let depth = file_depth f in
  let name = Filename.basename f.S.file_name in
  let ch = open_out (Filename.concat dir(name^".tex")) in
  let fmt = formatter_of_out_channel ch in
  latex_tabular_file n fmt depth provers f;
(*
    if table = "tabular" then
      begin
      latex_tabular n fmt depth provers t
      end
    else
      latex_longtable n fmt depth name provers t;

*)
    close_out ch



390 391 392 393 394 395 396 397 398 399 400 401 402 403 404
let element_latex_stat_goal g n table dir r =
  begin
    match r with
      | [] -> ()
      | _ ->
        eprintf "Warning: only main goals are supported@.:"
  end;
  standalone_goal_latex_stat n table dir g

let element_latex_stat_theory th n table dir e =
  match e with
    | [] -> theory_latex_stat n table dir th
    | g :: r ->
      try
        let goals =
Andrei Paskevich's avatar
Andrei Paskevich committed
405
          List.map (fun g -> (g.S.goal_name.Ident.id_string,g))
406 407 408 409 410 411 412 413 414
            th.S.theory_goals
        in
        let g = List.assoc g goals in
        element_latex_stat_goal g n table dir r
      with Not_found ->
        eprintf "Warning: goal not found in path: %s@." g

let element_latex_stat_file f n table dir e =
  match e with
415
    | [] -> file_latex_stat_all n table dir f
416 417 418 419 420 421 422 423 424 425 426 427 428 429
    | th :: r ->
      try
        let ths =
          List.map (fun th -> (th.S.theory_name.Ident.id_string,th))
            f.S.file_theories
        in
        let th = List.assoc th ths in
        element_latex_stat_theory th n table dir r
      with Not_found ->
        eprintf "Warning: theory not found in path: %s@." th


let element_latex_stat files n table dir e =
  eprintf "Element %s@." e;
430
  match Strings.split '.' e with
431 432
    | [] -> ()
    | f :: r ->
433 434 435 436
      let found = ref false in
      S.PHstr.iter
        (fun fname file ->
          let fname = Filename.basename fname in
437
          let fname = List.hd (Strings.split '.' fname) in
Andrei Paskevich's avatar
Andrei Paskevich committed
438
          if fname = f then
439 440 441 442 443 444 445
            begin
              found := true;
              element_latex_stat_file file n table dir r
            end)
        files;
      if not !found then
        eprintf "Warning: file not found for element %s@." e
446

447 448
let print_latex_statistics n table dir session =
  let files = session.S.session_files in
449 450 451 452 453
  match !opt_elements with
    | None ->
      S.PHstr.iter (fun _ f -> file_latex_stat n table dir f) files
    | Some l ->
      List.iter (element_latex_stat files n table dir) l
454 455 456 457 458 459 460



let table () = if !opt_longtable then "longtable" else "tabular"

let run_one fname =
  let project_dir = Session.get_project_dir fname in
461
  let session,_use_shapes = Session.read_session project_dir in
462 463 464 465
  let dir = if !opt_output_dir = "" then project_dir else
      !opt_output_dir
  in
  print_latex_statistics !opt_style (table ()) dir session
466 467

let run () =
MARCHE Claude's avatar
MARCHE Claude committed
468
  let _,_,should_exit1 = read_env_spec () in
469 470 471 472 473 474
  if should_exit1 then exit 1;
  iter_files run_one


let cmd =
  { cmd_spec = spec;
475
    cmd_desc = "output session in LaTeX format";
476
    cmd_name = "latex";
477 478
    cmd_run  = run;
  }