package mlpost

  1. Overview
  2. Docs
Legend:
Page
Library
Module
Module type
Parameter
Class
Class type
Source

Source file gentex.ml

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
(**************************************************************************)
(*                                                                        *)
(*  Copyright (C) Johannes Kanig, Stephane Lescuyer                       *)
(*  Jean-Christophe Filliatre, Romain Bardou and Francois Bobot           *)
(*                                                                        *)
(*  This software is free software; you can redistribute it and/or        *)
(*  modify it under the terms of the GNU Library General Public           *)
(*  License version 2.1, with the special exception on linking            *)
(*  described in file LICENSE.                                            *)
(*                                                                        *)
(*  This software 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.                  *)
(*                                                                        *)
(**************************************************************************)

open Point_lib

let jobname = "gentex"

let filename = File.from_string jobname

(* FIXME different from Metapost.default_prelude? *)

let latex_cmd tmpdir =
  Printf.sprintf "latex -jobname=%s -ipc -halt-on-error -output-dir '%s'"
    jobname
    (File.Dir.to_string tmpdir)

(*   Printf.sprintf "cat" *)

type t = {
  tex : Dviinterp.page;
  trans : Matrix.t;
  bb : float * float * float * float;
}

let set_verbosity _ = ()

type proc = { outc : in_channel; inc : out_channel; errc : in_channel }

type comm = { latex : proc; dvi : Dvi.Incremental.t; tmpdir : File.Dir.t }

let ends_with en s =
  let l = String.length s in
  let k = String.length en in
  if k <= l then
    try
      for i = 0 to k - 1 do
        if s.[l - i - 1] <> en.[k - i - 1] then raise Exit
      done;
      true
    with Exit -> false
  else false

let read_up_to_one =
  (* FIXME deal with EOF exception:
     Actually that print on stdout and that throw the error.
     Perhaps can we throw an exception with all in it.
     But it is not very fun for user (only part of the string in an exception
     is printed)
  *)
  let end_ = "[1]" in
  let rec aux buf inc =
    let s =
      try input_line inc
      with End_of_file ->
        Buffer.output_buffer stdout buf;
        invalid_arg "One tex snippet contains an error"
    in
    Buffer.add_string buf s;
    Buffer.add_char buf '\n';
    if ends_with end_ s then () else aux buf inc
  in
  fun inc ->
    let buf = Buffer.create 10 in
    aux buf inc

let read_up_to_one p = read_up_to_one p.outc

let mk_proc_latex tmpdir =
  let outc, inc, errc =
    Unix.open_process_full (latex_cmd tmpdir) (Unix.environment ())
  in
  { outc; inc; errc }

let write_to_proc p s = Printf.fprintf p.inc s

let push_prelude p prel =
  (* the shipout macro from metapost, that adds a vrule at the end of the tex
   * to obtain easily the size of the dvi *)
  write_to_proc p
    "%s\n\
     \\begin{document}\n\
     \\gdef\\mpxshipout{\\shipout\\hbox\\bgroup%%\n\
    \  \\setbox0=\\hbox\\bgroup}%%\n\
     \\gdef\\stopmpxshipout{\\egroup  \\dimen0=\\ht0 \\advance\\dimen0\\dp0\n\
    \  \\dimen1=\\ht0 \\dimen2=\\dp0\n\
    \  \\setbox0=\\hbox\\bgroup\n\
    \    \\box0\n\
    \    \\ifnum\\dimen0>0 \\vrule width1sp height\\dimen1 depth\\dimen2\n\
    \    \\else \\vrule width1sp height1sp depth0sp\\relax\n\
    \    \\fi\\egroup\n\
    \  \\ht0=0pt \\dp0=0pt \\box0 \\egroup}\n\
     %!"
    prel

let shipout_and_flush p s =
  write_to_proc p "\\mpxshipout %s\\stopmpxshipout\n%!" s

let extract cl =
  (* the vrule added by the metapost shipout macro is exploited and removed *)
  match cl with
  | Dviinterp.Fill_rect (_, x, y, _, h) :: cl ->
      let bb = (0., -.(y +. h), x, -.y) in
      { tex = cl; trans = Matrix.identity; bb }
  | _ -> assert false

let comm = ref None

(* TODO : do that only when clean is requested.
   note if an error occured comm become None
 *)
let () =
  at_exit (fun () ->
      match !comm with None -> () | Some p -> File.Dir.rm p.tmpdir)

let read_up_to_one p =
  try read_up_to_one p
  with e ->
    comm := None;
    raise e

let create tex =
  (* FIXME at some point we have to close the latex process *)
  match !comm with
  | None ->
      let tmpdir = Metapost_tool.create_temp_dir "mlpost" "" in
      let p = mk_proc_latex tmpdir in
      push_prelude p (Defaults.get_prelude ());
      shipout_and_flush p tex;
      read_up_to_one p;
      let filename = File.place tmpdir (File.set_ext filename "dvi") in
      let dvi_chan = File.open_in filename in
      let t, pgs = Dvi.Incremental.mk_t dvi_chan in
      comm := Some { latex = p; dvi = t; tmpdir };
      extract (Dviinterp.Incremental.load_page t (List.hd pgs))
  | Some p ->
      shipout_and_flush p.latex tex;
      read_up_to_one p.latex;
      let pgs = Dvi.Incremental.next_pages p.dvi in
      extract (Dviinterp.Incremental.load_page p.dvi (List.hd pgs))

let point_of_cm cm = 0.3937 *. 72. *. cm

let get_dimen_cm x = x.bb

(** donne la dimension en centimètre *)
let get_dimen_pt x =
  let x_min, y_min, x_max, y_max = get_dimen_cm x in
  (point_of_cm x_min, point_of_cm y_min, point_of_cm x_max, point_of_cm y_max)

let get_bases_cm _ = assert false

let get_bases_pt _ = assert false

let bounding_box x =
  let xmin, ymin, xmax, ymax = get_dimen_pt x in
  if Defaults.get_debug () then
    Format.printf "gentex bb : %f %f %f %f@." xmin ymin xmax ymax;
  ({ x = xmin; y = ymin }, { x = xmax; y = ymax })

let print fmt tex =
  let min, max = bounding_box tex in
  Format.fprintf fmt "[%a,%a]" print min print max

let deb_print fmt tex =
  Format.fprintf fmt "{ tex: %a ; matrix: %a }" Dviinterp.Print.page tex.tex
    Matrix.print tex.trans