package lambdapi

  1. Overview
  2. Docs
Proof assistant for the λΠ-calculus modulo rewriting

Install

dune-project
 Dependency

Authors

Maintainers

Sources

lambdapi-3.0.0.tbz
sha256=1066aed2618fd8e6a400c5147dbf55ea977ce8d3fe2e518ac6785c6775a1b8be
sha512=f7f499626aba92e070ae69581299a58525973fdbfd04a160ed3ac89209fb6cbe307b816d0b23e1b75bc83467ce8b4b0530c6f9816eaf58f7a07fde65a450106c

doc/src/lambdapi.core/env.ml.html

Source file env.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
(** Scoping environment for variables. *)

open Lplib open Base
open Term
open Common open Extra open Name
open Print

(** Type of an environment, used in scoping to associate names to
    corresponding variables and types. Note that it cannot be
    implemented by a map as the order is important. The structure is similar
    to {!type:Term.ctxt}, a tuple [(x,a,t)] is a variable [x], its type [a]
    and possibly its definition [t]. The typing environment [x1:A1,..,xn:An]
    is represented by the list [xn:An;..;x1:A1] in reverse order (last added
    variable comes first). *)
type env = (string * (var * term * term option)) list

type t = env

(** [pp ppf env] prints the environment [env] on the formatter [ppf] (for
   debug). *)
let pp : env pp =
  let def ppf t = out ppf " ≔ %a" term t in
  let elt ppf (s, (_,a,t)) = out ppf "%s: %a%a" s term a (Option.pp def) t in
  Common.Debug.D.list elt

(** [empty] is the empty environment. *)
let empty : env = []

(** [add n v a t env] extends the environment [env] by mapping the string
    [n] to [(v,a,t)]. *)
let add : string -> var -> term -> term option -> env -> env =
  fun n v a t env -> (n, (v, a, t)) :: env

(** [find n env] returns the variable associated to the variable  name
    [n] in the environment [env]. If none is found, [Not_found] is raised. *)
let find : string -> env -> var = fun n env ->
  let (x,_,_) = List.assoc n env in x

(** [mem n env] returns [true] iff [n] is mapped to a variable in [env]. *)
let mem : string -> env -> bool = List.mem_assoc

(** [to_prod env t] builds a sequence of products/lets whose domains are the
    variables of the environment [env] (from left to right), and whose body is
    the term [t]. By calling [to_prod [(xn,an,None);⋯;(x1,a1,None)] t] you
    obtain a term of the form [Πx1:a1,..,Πxn:an,t]. *)
let to_prod : env -> term -> term = fun env t ->
  let add_prod t (_,(x,a,d)) =
    let b = bind_var x t in
    match d with
    | None -> mk_Prod (a, b)
    | Some d -> mk_LLet (a, d, b)
  in
  List.fold_left add_prod t env

(** [to_abst env t] builds a sequence of abstractions or let bindings,
    depending on the definition of the elements in the environment whose
    domains are the variables of the environment [env] (from left to right),
    and which body is the term [t]:
    [to_abst [(xn,an,None);..;(x1,a1,None)] t = λx1:a1,..,λxn:an,t]. *)
let to_abst : env -> term -> term = fun env t ->
  let add_abst t (_,(x,a,d)) =
    let b = bind_var x t in
    match d with
    | None -> mk_Abst (a, b)
    | Some d -> mk_LLet (a, d, b)
  in
  List.fold_left add_abst t env

(** [vars env] extracts the array of variables in [env]. Note that the order
    is reversed: [vars [(xn,an);..;(x1,a1)] = [|x1;..;xn|]]. *)
let vars : env -> var array = fun env ->
  Array.of_list (List.rev_map (fun (_,(x,_,_)) -> x) env)

(** [appl t env] applies [t] to the variables of [env]. *)
let appl : term -> env -> term = fun t env ->
  List.fold_right (fun (_,(x,_,_)) t -> mk_Appl (t, mk_Vari x)) env t

(** [to_terms env] extracts the array of the variables in [env] and injects
    them in [tbox]. This is the same as [Array.map mk_Vari (vars env)]. Note
    that the order is reversed: [to_tbox [(xn,an);..;(x1,a1)] =
    [|x1;..;xn|]]. *)
let to_terms : env -> term array = fun env ->
  Array.of_list (List.rev_map (fun (_,(x,_,_)) -> mk_Vari x) env)

(** [to_ctxt e] converts an environment into a context. *)
let to_ctxt : env -> ctxt = List.map snd

(** [match_prod c t f] returns [f a None b] if [t] matches [Prod(a,b)],
    and [f a d b] if [t] matches [LLet(a,d,b)], possibly after reduction.
@raise [Invalid_argument] if the whnf of [t] is not a product. *)
let match_prod : ctxt -> term -> (term -> term option -> binder -> 'a) -> 'a
  = fun c t f ->
  match unfold t with
  | Prod(a,b) -> f a None b
  | LLet(a,d,b) -> f a (Some d) b
  | _ ->
      match Eval.whnf c t with
      | Prod(a,b) -> f a None b
      | _ -> invalid_arg __LOC__

(** [of_prod c s t] returns a tuple [(env,b)] where [b] is constructed from
   the term [t] by unbinding as much dependent products as possible in the
   head of [t]. The free variables created by this process, prefixed by [s],
   are given (with their types) in the environment [env] (in reverse
   order). For instance, if [t] is of the form [Πx1:a1, ⋯, Πxn:an, b], then
   the function returns [b] and the environment [(xn,an); ⋯;(x1,a1)]. *)
let of_prod : ctxt -> string -> term -> env * term = fun c s t ->
  let i = Stdlib.ref (-1) in
  let rec build_env env t =
    try match_prod c t (fun a d b ->
            let name = Stdlib.(incr i; s ^ string_of_int !i) in
            let x, b = unbind ~name b in
            build_env (add name x a d env) b)
    with Invalid_argument _ -> env, t
  in build_env [] t

(** [of_prod_nth c n t] returns a tuple [(env,b)] where [b] is constructed
   from the term [t] by unbinding [n] dependent products. The free variables
   created by this process are given (with their types) in the environment
   [env] (in reverse order). For instance, if [t] is of the form [Πx1:a1, ⋯,
   Πxn:an, b] then the function returns [b] and the environment [(xn,an);
   ⋯;(x1,a1)]. [n] must be non-negative.
@raise [Invalid_argument] if [t] does not evaluate to a series of (at least)
   [n] products. *)
let of_prod_nth : ctxt -> int -> term -> env * term = fun c n t ->
  let rec build_env i env t =
    if i >= n then env, t
    else match_prod c t (fun a d b ->
             let x, b = unbind b in
             build_env (i+1) (add (base_name x) x a d env) b)
  in build_env 0 [] t

(** [of_prod_using c xs t] is similar to [of_prod s c n t] where [n =
   Array.length xs] except that it replaces unbound variables by those of
   [xs].
@raise [Invalid_argument] if [t] does not evaluate to a series of (at least)
   [n] products. *)
let of_prod_using : ctxt -> var array -> term -> env * term = fun c xs t ->
  let n = Array.length xs in
  let rec build_env i env t =
    if i >= n then env, t
    else match_prod c t (fun a d b ->
             let xi = xs.(i) in
             let name = base_name xi in
             let env = add name xi a d env in
             build_env (i+1) env (subst b (mk_Vari xi)))
  in build_env 0 [] t

(** [names env] returns the set of names in [env]. *)
let names : env -> int StrMap.t =
  let add_decl idmap (s,_) = add_name s idmap in
  List.fold_left add_decl StrMap.empty

(** [gen_valid_idopts env ids] generates a list of pairwise distinct
    identifiers distinct from those of [env] to replace [ids]. *)
let gen_valid_idopts (env:env) (ids: string list): Pos.strloc option list =
  let f id (idopts, idmap) =
    let id, idmap = get_safe_prefix id idmap in
    Some(Pos.none id)::idopts, idmap
  in
  fst (List.fold_right f ids ([], names env))