Source file dk.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
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
(** Export a Lambdapi signature to Dedukti. *)
open Lplib open Base open Extra
open Timed
open Common
open Core open Term
let string = string
(** Translation of identifiers. Lambdapi identifiers that are Dedukti keywords
   or invalid Dedukti identifiers are escaped, a feature offered by
   Dedukti. *)
let keyword_table = Hashtbl.create 59
let is_keyword : string -> bool = Hashtbl.mem keyword_table
let _ = let open Parsing.DkTokens in
  let loc = Lexing.dummy_pos, Lexing.dummy_pos in
  List.iter (fun (s, t) -> Hashtbl.add keyword_table s t)
    [ ".", DOT
    ; ",", COMMA
    ; ":", COLON
    ; "==", EQUAL
    ; "[", LEFTSQU
    ; "]", RIGHTSQU
    ; "{", LEFTBRA
    ; "}", RIGHTBRA
    ; "(", LEFTPAR
    ; ")", RIGHTPAR
    ; "-->", LONGARROW
    ; "->", ARROW
    ; "=>", FATARROW
    ; ":=", DEF
    ; "Type", TYPE loc
    ; "def", KW_DEF loc
    ; "defac", KW_DEFAC loc
    ; "defacu", KW_DEFACU loc
    ; "injective", KW_INJ loc
    ; "thm", KW_THM loc
    ; "private", KW_PRV loc
    ; "#NAME", NAME (loc, "")
    ; "#REQUIRE", REQUIRE (loc, "")
    ; "#EVAL", EVAL loc
    ; "#INFER", INFER loc
    ; "#CHECK", CHECK loc
    ; "#CHECKNOT", CHECKNOT loc
    ; "#ASSERT", ASSERT loc
    ; "#ASSERTNOT", ASSERTNOT loc
    ; "#PRINT", PRINT loc
    ; "#GDT", GDT loc]
let is_ident : string -> bool = fun s ->
  Parsing.DkLexer.is_ident (Lexing.from_string s)
let is_mident : string -> bool = fun s ->
  Parsing.DkLexer.is_mident (Lexing.from_string s)
let escape : string pp = fun ppf s -> out ppf "{|%s|}" s
let replace_spaces : string -> string = fun s ->
  let open Bytes in
  let b = of_string s in
  for i=0 to length b - 1 do
    match get b i with
    | ' ' | '\n' -> set b i '_'
    | _ -> ()
  done;
  to_string b
let ident : string pp = fun ppf s ->
  if s = "" then escape ppf s
  else if s.[0] = '{' then string ppf (replace_spaces s)
  else if is_keyword s then escape ppf s
  else if is_ident s then string ppf s
  else escape ppf s
(** Translation of paths. Paths equal to the [!current_path] are not
   printed. Non-empty paths end with a dot. We assume that the module p1.p2.p3
   is in the file p1_p2_p3.dk. *)
let path_elt : string pp = fun ppf s ->
  if s <> "" && s.[0] = '{' then
    string ppf (replace_spaces (Escape.unescape s))
  else string ppf s
let current_path = Stdlib.ref []
let path : Path.t pp = fun ppf p ->
  if p <> Stdlib.(!current_path) then
  match p with
  | [] -> ()
  | p ->
      let joined_path = Format.asprintf "%a" (List.pp path_elt "_") p in
      if List.for_all is_mident p then
        out ppf "%s." joined_path
      else
        Format.fprintf ppf "%a." escape joined_path
let qid : (Path.t * string) pp = fun ppf (p, i) ->
  out ppf "%a%a" path p ident i
(** Type of Dedukti declarations. *)
type decl =
  | Sym of sym
  | Rule of (Path.t * string * rule)
(** Declarations are ordered wrt their positions in the source. *)
let pos_of_decl : decl -> Pos.popt = fun i ->
  match i with
  | Sym s -> s.sym_pos
  | Rule (_,_,r) -> r.rule_pos
let cmp : decl cmp = cmp_map (Lplib.Option.cmp Pos.cmp) pos_of_decl
(** Translation of terms. *)
let tvar : tvar pp = fun ppf v -> ident ppf (Bindlib.name_of v)
let tevar : tevar pp = fun ppf v -> out ppf "_%a" ident (Bindlib.name_of v)
let patt : string pp = fun ppf v -> out ppf "_%s" v
let tenv : term_env pp = fun ppf te ->
  match te with
  | TE_Vari v -> tevar ppf v
  | TE_Some _ -> assert false
  | TE_None -> assert false
let rec term : bool -> term pp = fun b ppf t ->
  match unfold t with
  | Vari v -> tvar ppf v
  | Type -> out ppf "Type"
  | Kind -> assert false
  | Symb s -> qid ppf (s.sym_path, s.sym_name)
  | Prod(t,u) ->
    let x,u' = Bindlib.unbind u in
    if Bindlib.binder_constant u
    then out ppf "(%a -> %a)" (term b) t (term b) u'
    else out ppf "(%a : %a -> %a)" tvar x (term b) t (term b) u'
  | Abst(t,u) ->
    let x,u = Bindlib.unbind u in
    if b then out ppf "(%a : %a => %a)" tvar x (term b) t (term b) u
    else out ppf "(%a => %a)" tvar x (term b) u
  | Appl _ ->
    let h, ts = get_args t in
    out ppf "(%a%a)" (term b) h (List.pp (prefix " " (term b)) "") ts
  | LLet(a,t,u) ->
    let x,u = Bindlib.unbind u in
    out ppf "((%a : %a := %a) => %a)" tvar x (term b) a (term b) t (term b) u
  | Patt(_,s,[||]) -> patt ppf s
  | Patt(_,s,ts) ->
    out ppf "(%a%a)" patt s (Array.pp (prefix " " (term b)) "") ts
  | TEnv(te, [||]) -> tenv ppf te
  | TEnv(te, ts) ->
    out ppf "(%a%a)" tenv te (Array.pp (prefix " " (term b)) "") ts
  | TRef _ -> assert false
  | Wild -> assert false
  | Meta _ -> assert false
  | Plac _ -> assert false
(** Translation of declarations. *)
let modifiers : sym -> string list = fun s ->
  let open Stdlib in
  let r = ref [] in
  let add m = r := m::!r in
  begin
    match s.sym_prop with
    | Const -> ()
    | Injec -> add "injective"
    | AC _ -> add "defac"
    | Defin -> add "def"
    | Assoc _ -> assert false
    | Commu -> assert false
  end;
  if s.sym_expo = Protec then add "private";
  !r
let sym_decl : sym pp = fun ppf s ->
  match !(s.sym_def) with
  | None ->
    begin match s.sym_prop with
      | AC _ ->
        begin match unfold !(s.sym_type) with
          | Prod(t,_) ->
            out ppf "%a%a [%a].@."
              (List.pp (suffix string " ") "") (modifiers s)
              ident s.sym_name (term true) t
          | _ -> assert false
        end
      | _ ->
        out ppf "%a%a : %a.@."
          (List.pp (suffix string " ") "") (modifiers s)
          ident s.sym_name (term true) !(s.sym_type)
    end
  | Some d ->
    if !(s.sym_opaq) then
      out ppf "thm %a : %a := %a.@."
        ident s.sym_name (term true) !(s.sym_type) (term true) d
    else
      out ppf "%a%a : %a := %a.@."
        (List.pp (suffix string " ") "") (modifiers s)
        ident s.sym_name (term true) !(s.sym_type) (term true) d
let rule_decl : (Path.t * string * rule) pp = fun ppf (p, n, r) ->
  let xs, rhs = Bindlib.unmbind r.rhs in
  out ppf "[%a] %a%a --> %a.@."
    (Array.pp tevar ", ") xs qid (p, n)
    (List.pp (prefix " " (term false)) "") r.lhs (term true) rhs
let decl : decl pp = fun ppf decl ->
  match decl with
  | Sym s -> sym_decl ppf s
  | Rule r -> rule_decl ppf r
(** [decls_of_sign sign] computes a list of declarations for the
   signature [sign], in order of appearance in the source. *)
let decls_of_sign : Sign.t -> decl list = fun sign ->
  let add_sym l s = List.insert cmp (Sym s) l
  and add_rule p n l r =
    if p = Ghost.sign.sign_path
    then l
    else List.insert cmp (Rule (p, n, r)) l
  in
  let add_sign_symbol n s l =
    List.fold_left (add_rule [] n) (add_sym l s) !(s.sym_rules) in
  let add_rules p n rs l = List.fold_left (add_rule p n) l rs in
  let add_sign_dep p map l = StrMap.fold (add_rules p) map l in
  StrMap.fold add_sign_symbol !(sign.sign_symbols)
    (Path.Map.fold add_sign_dep !(sign.sign_deps) [])
(** Translation of a signature. *)
let require : Path.t -> _ -> unit = fun p _ ->
  if p <> Ghost.sign.sign_path then Format.printf "#REQUIRE %a@." path p
let sign : Sign.t -> unit = fun sign ->
  Path.Map.iter require !(sign.sign_deps);
  Stdlib.(current_path := sign.sign_path);
  List.iter (decl Format.std_formatter) (decls_of_sign sign)