package bindlib
Legend:
Page
Library
Module
Module type
Parameter
Class
Class type
Source
Page
Library
Module
Module type
Parameter
Class
Class type
Source
Source file bindlib.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 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311 312 313 314 315 316 317 318 319 320 321 322 323 324 325 326 327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345 346 347 348 349 350 351 352 353 354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 387 388 389 390 391 392 393 394 395 396 397 398 399 400 401 402 403 404 405 406 407 408 409 410 411 412 413 414 415 416 417 418 419 420 421 422 423 424 425 426 427 428 429 430 431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 458 459 460 461 462 463 464 465 466 467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 496 497 498 499 500 501 502 503 504 505 506 507 508 509 510 511 512 513 514 515 516 517 518 519 520 521 522 523 524 525 526 527 528 529 530 531 532 533 534 535 536 537 538 539 540 541 542 543 544 545 546 547 548 549 550 551 552 553 554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 588 589 590 591 592 593 594 595 596 597 598 599 600 601 602 603 604 605 606 607 608 609 610 611 612 613 614 615 616 617 618 619 620 621 622 623 624 625 626 627 628 629 630 631 632 633 634 635 636 637 638 639 640 641 642 643 644 645 646 647 648 649 650 651 652 653 654 655 656 657 658 659 660 661 662 663 664 665 666 667 668 669 670 671 672 673 674 675 676 677 678 679 680 681 682 683 684 685 686 687 688 689 690 691 692 693 694 695 696 697 698 699 700 701 702 703 704 705 706 707 708 709 710 711 712 713 714 715 716 717 718 719 720 721 722 723 724 725 726 727 728 729 730 731 732 733 734 735 736 737 738 739 740 741 742 743 744 745 746 747 748 749 750 751 752 753 754 755 756 757 758 759 760 761 762 763 764 765 766 767 768 769 770 771
(** The [Bindlib] library provides support for free and bound variables in the OCaml language. The main application is the representation of types with a binding structure (e.g., abstract syntax trees). @author Christophe Raffalli @author Rodolphe Lepigre @version 6.0.0 *) (* Counter for generating fresh variable keys (i.e., unique identifiers). *) let ((reset_counter : unit -> unit), (fresh_key : unit -> int)) = let c = ref (-1) in ((fun () -> c := -1), (fun () -> incr c; !c)) (* FIXME remove in recent enough OCaml. *) module Int = struct type t = int let compare = (-) end module IMap = Map.Make(Int) module SMap = Map.Make(String) (* Environment **************************************************************) (* The [Bindlib] library uses environments to store values associated to bound variables upon substitution. We rely on the [Obj] module to store variables with potentially different types in a single array. Uses of the [Env] module by [Bindlib] are safe because every single cell of every environment is only used at a single, fixed type. Moreover, there can be no problem with unboxed floating-point arrays because it is not possible to extend the [float] type itself with variables. *) module Env : sig (* Type of an environment. *) type t (* [create next_free size] creates a new empty environment of with the given [size], and with its next free cell at index [next_free]. *) val create : int -> int -> t (* [set env i e] sets the value stored at position [i] of [env] to [e]. *) val set : t -> int -> 'a -> unit (* [get i env] gets the value stored at position [i] in [env]. *) val get : int -> t -> 'a (* [blit s t n] copies the [n] first elements of [s] to [t]. *) val blit : t -> t -> int -> unit (* [copy env] make a copy of environment [env]. *) val copy : t -> t (* [get_next_free env] returns the index of the next free cell in [env]. *) val get_next_free : t -> int (* [set_next_free env i] sets the next free cell index of [env] to [i]. *) val set_next_free : t -> int -> unit end = struct type t = {tab : Obj.t array; mutable next_free : int} let create next_free size = {tab = Array.make size (Obj.repr ()); next_free} let set env i e = Array.set env.tab i (Obj.repr e) let get i env = Obj.obj (Array.get env.tab i) let blit src dst len = Array.blit src.tab 0 dst.tab 0 len let copy env = {tab = Array.copy env.tab; next_free = env.next_free} let get_next_free env = env.next_free let set_next_free env n = env.next_free <- n end (* Closure representation ***************************************************) (* In the internals of [Bindlib], variables are identified by a unique integer key. Closures are then formed by mapping each free variable to a slot in an environment (of type [Env.t]), but not directly. The mapping is established using a map (of type [varpos]) which associates variable keys to indices in the environment. *) type varpos = int IMap.t type 'a closure = varpos -> Env.t -> 'a let map_closure_aux f a = fun env -> f (a env) let map_closure : ('a -> 'b) -> 'a closure -> 'b closure = fun f cla vs -> map_closure_aux f (cla vs) let app_closure_aux f a = fun env -> f env a let app_closure : ('a -> 'b) closure -> 'a -> 'b closure = fun clf a vs -> app_closure_aux (clf vs) a let apply_closure_aux f a = fun env -> f env (a env) let (<*>) : ('a -> 'b) closure -> 'a closure -> 'b closure = fun clf cla vs -> apply_closure_aux (clf vs) (cla vs) (* Box and variable representation ******************************************) (* At the core of [Bindlib] is the type ['a box] which represents an object of type ['a] whose free variables are available for binding. Intuitively, this is represented as the combination of an environment and a closure as in the [Env(vs,n,t)]. The [Box(e)] constructor's only role is to optimise the case where an object is known to be closed. In the [Env(vs,n,t)] constructor, the list [vs] contains all free variables in scope (sorted by their keys), the integer [n] counts the number of bound variables that have a reserved slot in the environment, and finally, [t] is a representation of the objects (with variables) using a closure. Note that the closure [t] is intended to receive the environment as second argument, and the position of the free variables of [vs] in the environment as a first argument. Important remark: to be able to efficiently implement substitution, we make sure that the [varpos] map (given as first argument to closures) is used at most once for each variable, even if a variable appears many times. *) type 'a box = | Box of 'a (* Closed object. *) | Env of any_var list * int * 'a closure (* Open object with environment. *) and 'a var = { var_key : int; (* Unique identifier. *) var_name : string; (* Name as a free variable. *) var_mkfree : 'a var -> 'a; (* Function to build a term. *) var_box : 'a box; (* Variable as a boxed object. *) } (* Variable of any type (using an existential). In principle, this constructor can be unboxed since it has a single constructor. However, this only works starting from OCaml 4.11.0. The annotation is erased for prior versions. *) and any_var = V : 'a var -> any_var [@@unboxed] let name_of x = x.var_name let hash_var x = Hashtbl.hash (`HVar, x.var_key) let box_var x = x.var_box let compare_vars x y = y.var_key - x.var_key let eq_vars x y = x.var_key = y.var_key let uid_of x = x.var_key type 'a mvar = 'a var array let names_of xs = Array.map name_of xs let uids_of xs = Array.map uid_of xs (* The [merge_uniq] and [remove] functions manipulate variables lists found in boxed object of the form [Env(_,_,_)]. They rely on the fact that such list is always sorted in the order of variable keys, and has no duplicates. *) let merge_uniq : any_var list -> any_var list -> any_var list = let rec merge_uniq acc l1 l2 = match (l1, l2) with | ([] , _ ) -> List.rev_append acc l2 | (_ , [] ) -> List.rev_append acc l1 | ((V(x) as vx)::xs, (V(y) as vy)::ys) -> if x.var_key = y.var_key then merge_uniq (vx::acc) xs ys else if x.var_key < y.var_key then merge_uniq (vx::acc) xs l2 else (* x.var_key > y.var_key*) merge_uniq (vy::acc) l1 ys in merge_uniq [] let remove : 'a var -> any_var list -> any_var list = fun {var_key ; _} -> let rec remove acc = function | (V(x) as v)::l when x.var_key < var_key -> remove (v::acc) l | (V(x) )::l when x.var_key = var_key -> List.rev_append acc l | _ -> raise Not_found in remove [] (* Function [minimize vs n cl] constructs a minimal closure equivalent to [cl] and only containing variables of [vs]. Additionally, the function builds an environment with [n] extra slots. *) let minimize_aux_prefix size n t = fun env -> let new_env = Env.create size (size + n) in Env.blit env new_env size; t new_env let minimize_aux tab n t = fun env -> let size = Array.length tab in let new_env = Env.create size (size + n) in Array.iteri (fun i x -> Env.set new_env i (Env.get x env)) tab; t new_env let minimize : any_var list -> int -> 'a closure -> 'a closure = fun vs n t -> if n = 0 then t else fun vp -> let size = List.length vs in let tab = Array.make size 0 in let prefix = ref true in let f (new_vp, i) (V(var)) = let j = IMap.find var.var_key vp in if i <> j then prefix := false; tab.(i) <- j; (IMap.add var.var_key i new_vp, i+1) in let (new_vp,_) = List.fold_left f (IMap.empty,0) vs in let t = t new_vp in if !prefix then minimize_aux_prefix size n t else minimize_aux tab n t let box : 'a -> 'a box = fun t -> Box (t) let apply_box : ('a -> 'b) box -> 'a box -> 'b box = fun f a -> match (f, a) with | (Box(f) , Box(a) ) -> Box(f a) | (Box(f) , Env(va,na,ta)) -> Env(va, na, map_closure f ta) | (Env(vf,nf,tf), Box(a) ) -> Env(vf, nf, app_closure tf a) | (Env(vf,nf,tf), Env(va,na,ta)) -> Env(merge_uniq vf va, 0, minimize vf nf tf <*> minimize va na ta) let occur : 'a var -> 'b box -> bool = fun v b -> match b with | Box(_) -> false | Env(vs,_,_) -> List.exists (fun (V(x)) -> x.var_key = v.var_key) vs let is_closed : 'a box -> bool = fun b -> match b with Box(_) -> true | Env(_,_,_) -> false let box_apply : ('a -> 'b) -> 'a box -> 'b box = fun f a -> match a with | Box(a) -> Box(f a) | Env(vs,na,ta) -> Env(vs, na, map_closure f ta) let box_apply2 f ta tb = apply_box (box_apply f ta) tb let box_apply3 f ta tb tc = apply_box (box_apply2 f ta tb) tc let box_apply4 f ta tb tc td = apply_box (box_apply3 f ta tb tc) td let box_pair x y = box_apply2 (fun x y -> (x,y)) x y let box_triple x y z = box_apply3 (fun x y z -> (x,y,z)) x y z let box_opt o = match o with | None -> box None | Some(e) -> box_apply (fun e -> Some(e)) e let dummy_box : 'a box = let fail _ = failwith "Invalid use of dummy_box" in Env([], 0, fail) let unbox b = match b with Box(t) -> t | Env(vs,nb,t) -> let nbvs = List.length vs in let env = Env.create nbvs (nbvs + nb) in let cur = ref 0 in let fn vp (V(x)) = let i = !cur in incr cur; Env.set env i (x.var_mkfree x); IMap.add x.var_key i vp in t (List.fold_left fn IMap.empty vs) env (* Functorial interface for boxing functions ********************************) (* We use the following type and functions for combining several boxed objects into one. In particular, we combine all lists of variables [vs] that appear in the encountered [Env(vs,n,_)] constructors. We also remember the largest [n] encountered in such constructors, and whether we encountered only [Box] constructors (i.e., closed objects). The type [data] is used to represent these three pieces of information. The function [gather_data data b] combines data gathered so far (in [data]) and new data from a boxed object [b]. Note that [no_data] plays the role of the neutral element of [gather_data]. *) type data = bool * any_var list * int let gather_data : data -> 'a box -> data = fun data b -> match (b, data) with | (Box(_) , _ ) -> data | (Env(vs,n,_), (_,ws,m)) -> (false, merge_uniq ws vs, max m n) let no_data : data = (true, [], 0) (* Function [box_to_closure b] constructs a closure from boxed object [b]. The information about variables and reserved slots (contained in boxed objects) is lost in the process, and it must thus be tracked independently (with the use of [gather_data]) to reconstruct a boxed object in the end. *) let box_to_closure : 'a box -> 'a closure = fun b -> match b with | Box(t) -> fun _ _ -> t | Env(vs,n,t) -> minimize vs n t module type Map = sig type 'a t val map : ('a -> 'b) -> 'a t -> 'b t end module Lift(M : Map) = struct let lift_box_aux m = fun env -> M.map (fun o -> o env) m let lift_box : 'a box M.t -> 'a M.t box = fun m -> let data = ref no_data in let fn b = data := gather_data !data b; box_to_closure b in let m = M.map fn m in let aux vp = let m = M.map (fun o -> o vp) m in lift_box_aux m in match !data with | (true, _ , _) -> Box(aux IMap.empty (Env.create 0 0)) | (_ , vs, n) -> Env(vs, n, minimize vs n aux) end module type Map2 = sig type ('a, 'b) t val map : ('a -> 'b) -> ('c -> 'd) -> ('a, 'c) t -> ('b, 'd) t end module Lift2(M : Map2) = struct let lift_box_aux m = fun env -> M.map (fun o -> o env) (fun o -> o env) m let lift_box : ('a box, 'b box) M.t -> ('a,'b) M.t box = fun m -> let data = ref no_data in let fn b = data := gather_data !data b; box_to_closure b in let m = M.map fn fn m in let aux vp = let m = M.map (fun o -> o vp) (fun o -> o vp) m in lift_box_aux m in match !data with | (true, _ , _) -> Box(aux IMap.empty (Env.create 0 0)) | (_ , vs, n) -> Env(vs, n, minimize vs n aux) end module Lift_list = Lift( struct type 'a t = 'a list let map = List.map end) let box_list = Lift_list.lift_box module Lift_rev_list = Lift( struct type 'a t = 'a list let map = List.rev_map end) let box_rev_list = Lift_rev_list.lift_box module Lift_array = Lift( struct type 'a t = 'a array let map = Array.map end) let box_array = Lift_array.lift_box (* Representation of binders ************************************************) type ('a,'b) binder = { b_name : string; (* Preferred name for the bound variable. *) b_bind : bool; (* Indicates whether the variable occurs. *) b_rank : int; (* Number of remaining free variables. *) b_mkfree : 'a var -> 'a; (* Injection of variables into domain. *) b_value : 'a -> 'b; (* Substitution function. *) } let binder_name b = b.b_name let subst b x = b.b_value x let binder_occur b = b.b_bind let binder_constant b = not b.b_bind let binder_closed b = b.b_rank = 0 let binder_rank b = b.b_rank let binder_compose b f = {b with b_value = (fun x -> f (b.b_value x))} type ('a,'b) mbinder = { mb_names : string array; (* Preferred names for the bound variables. *) mb_binds : bool array; (* Indicates whether the variables occur. *) mb_rank : int; (* Number of remaining free variables. *) mb_mkfree : 'a var -> 'a; (* Injection of variables into domain. *) mb_value : 'a array -> 'b; (* Substitution function. *) } let mbinder_arity b = Array.length b.mb_names let mbinder_names b = b.mb_names let msubst b xs = b.mb_value xs let mbinder_occurs b = b.mb_binds let mbinder_constant b = not (Array.fold_left (||) false b.mb_binds) let mbinder_closed b = b.mb_rank = 0 let mbinder_rank b = b.mb_rank let mbinder_compose b f = {b with mb_value = (fun x -> f (b.mb_value x))} let raw_binder b_name b_bind b_rank b_mkfree b_value = {b_name; b_bind; b_rank; b_mkfree; b_value} let raw_mbinder mb_names mb_binds mb_rank mb_mkfree mb_value = {mb_names; mb_binds; mb_rank; mb_mkfree; mb_value} let bind_apply b arg = box_apply2 subst b arg let mbind_apply b args = box_apply2 msubst b args (* Variable creation ********************************************************) let build_var_aux key vp = Env.get (IMap.find key vp) let build_var var_key var_mkfree name = let rec x = let var_box = Env([V(x)], 0, build_var_aux var_key) in {var_key; var_name = name; var_mkfree; var_box} in x let new_var mkfree name = build_var (fresh_key ()) mkfree name let new_mvar mkfree names = Array.map (fun n -> new_var mkfree n) names let copy_var x mkfree = build_var x.var_key mkfree let unbind b = let x = new_var b.b_mkfree (binder_name b) in (x, subst b (b.b_mkfree x)) let unbind2 b1 b2 = let x = new_var b1.b_mkfree (binder_name b1) in let v = b1.b_mkfree x in (x, subst b1 v, subst b2 v) let eq_binder eq f g = f == g || let (_,t,u) = unbind2 f g in eq t u let unmbind b = let x = new_mvar b.mb_mkfree (mbinder_names b) in (x, msubst b (Array.map b.mb_mkfree x)) let unmbind2 b1 b2 = if mbinder_arity b1 <> mbinder_arity b2 then invalid_arg "Arity mismatch in unmbind2"; let xs = new_mvar b1.mb_mkfree (mbinder_names b1) in let vs = Array.map b1.mb_mkfree xs in (xs, msubst b1 vs, msubst b2 vs) let eq_mbinder eq f g = f == g || (mbinder_arity f = mbinder_arity g && let (_,t,u) = unmbind2 f g in eq t u) (* Implementation of [bind_var] *********************************************) let build_binder x b_rank b_bind b_value = {b_name = x.var_name; b_rank; b_bind; b_value; b_mkfree = x.var_mkfree} let bind_var_aux1 n t = fun arg -> let env = Env.create 1 (n+1) in Env.set env 0 arg; t env let bind_var_aux2 rank t = fun env arg -> let next = Env.get_next_free env in if next = rank then begin Env.set_next_free env (next + 1); Env.set env next arg; t env end else begin let env = Env.copy env in Env.set_next_free env (rank+1); for i = rank+1 to next-1 do Env.set env i 0 done; Env.set env rank arg; t env end let bind_var_aux3 x rank t = fun env -> let value = bind_var_aux2 rank t env in build_binder x rank true value let bind_var_aux4 t = fun env _ -> t env let bind_var_aux5 x rank t = fun env -> let value = bind_var_aux4 t env in build_binder x rank false value let bind_var : 'a var -> 'b box -> ('a, 'b) binder box = fun x b -> match b with | Box(t) -> Box(build_binder x 0 false (fun _ -> t)) | Env(vs,n,t) -> try match vs with | [V(y)] -> if x.var_key <> y.var_key then raise Not_found; (* The variable to bind is the last one. *) let r = 0 in let t = t (IMap.singleton x.var_key r) in let value = bind_var_aux1 n t in Box(build_binder x 0 true value) | _ -> let vs = remove x vs in (* General case. *) let cl vp = let rank = List.length vs in let r = rank in let t = t (IMap.add x.var_key r vp) in bind_var_aux3 x rank t in Env(vs, n+1, cl) with Not_found -> (* The variable does not occur. *) let value vp = let t = t vp in let rank = List.length vs in bind_var_aux5 x rank t in Env(vs, n, value) let box_binder f b = if binder_closed b then box b else let (x,t) = unbind b in bind_var x (f t) (* Implementation of [bind_mvar] ********************************************) let check_arity : 'a mvar -> 'a array -> unit = fun xs args -> if Array.(length xs <> length args) then invalid_arg "Bad arity in msubst" let bind_mvar_aux0 xs t = fun args -> check_arity xs args; t let bind_mvar_aux1 m xs mb_binds t = fun args -> check_arity xs args; let v = Env.create 0 m in let pos = ref 0 in for i = 0 to Array.length xs - 1 do if mb_binds.(i) then begin Env.set v !pos args.(i); incr pos; end done; Env.set_next_free v !pos; t v let bind_mvar_aux2 xs t = fun env args -> check_arity xs args; t env let bind_mvar_aux3 xs t mb_names mb_rank mb_binds mb_mkfree = fun env -> let mb_value = bind_mvar_aux2 xs t env in {mb_names; mb_rank; mb_binds; mb_value; mb_mkfree} let bind_mvar_aux4 xs t mb_rank mb_binds = fun env args -> check_arity xs args; let next = Env.get_next_free env in let cur_pos = ref mb_rank in if next = mb_rank then begin for i = 0 to Array.length xs - 1 do if mb_binds.(i) then begin Env.set env !cur_pos args.(i); incr cur_pos; end done; Env.set_next_free env !cur_pos; t env end else begin let env = Env.copy env in for i = 0 to Array.length xs - 1 do if mb_binds.(i) then begin Env.set env !cur_pos args.(i); incr cur_pos; end done; Env.set_next_free env !cur_pos; for i = !cur_pos to next - 1 do Env.set env i 0 done; t env end let bind_mvar_aux5 xs t mb_names mb_rank mb_binds mb_mkfree = fun env -> let mb_value = bind_mvar_aux4 xs t mb_rank mb_binds env in {mb_names; mb_rank; mb_binds; mb_value; mb_mkfree} let bind_mvar : 'a mvar -> 'b box -> ('a,'b) mbinder box = fun xs b -> let mb_mkfree = if Array.length xs > 0 then xs.(0).var_mkfree else (fun _ -> assert false) in match b with | Box(t) -> let mb_binds = Array.map (fun _ -> false) xs in let mb_names = Array.map name_of xs in let mb_value = bind_mvar_aux0 xs t in Box({mb_names; mb_rank = 0; mb_binds; mb_value; mb_mkfree}) | Env(vs,n,t) -> let keys = Array.map (fun _ -> 0) xs in let vss = Array.map (fun _ -> vs) xs in let (vs, m) = let vs = ref vs in let m = ref n in for i = Array.length xs - 1 downto 0 do let v = xs.(i) in begin try vs := remove v !vs; incr m; keys.(i) <- v.var_key with Not_found -> keys.(i) <- -1 end; vss.(i) <- !vs (*NOTE: store each vs, for good renaming *) done; (!vs, !m) in if vs = [] then (* All the free variables become bound. *) let mb_names = Array.map (fun _ -> "") xs in let cur_pos = ref 0 in let vp = ref IMap.empty in let f i key = mb_names.(i) <- xs.(i).var_name; if key >= 0 then begin vp := IMap.add key !cur_pos !vp; incr cur_pos; true end else false in let mb_binds = Array.mapi f keys in let t = t !vp in let mb_value = bind_mvar_aux1 m xs mb_binds t in Box({mb_names; mb_binds; mb_rank = 0; mb_value; mb_mkfree}) else if m = n then (* None of the variables occur. *) let cl vp = let mb_rank = List.length vs in let mb_binds = Array.map (fun _ -> false) xs in let fn x = x.var_name in let mb_names = Array.map fn xs in let t = t vp in bind_mvar_aux3 xs t mb_names mb_rank mb_binds mb_mkfree in Env(vs, n, cl) else (* General case. *) let cl vp = let mb_names = Array.map (fun _ -> "") xs in let mb_rank = List.length vs in let cur_pos = ref mb_rank in let vp = ref vp in let f i key = mb_names.(i) <- xs.(i).var_name; if key >= 0 then (vp := IMap.add key !cur_pos !vp; incr cur_pos; true) else false in let mb_binds = Array.mapi f keys in let t = t !vp in bind_mvar_aux5 xs t mb_names mb_rank mb_binds mb_mkfree in Env(vs, m, cl) let box_mbinder f b = if b.mb_rank = 0 then box b else let (xs,t) = unmbind b in bind_mvar xs (f t) (* Functorial interface for renaming policies *******************************) module type Renaming = sig type ctxt val empty_ctxt : ctxt val reset_context_for_closed_terms : bool val skip_constant_binders : bool val constant_binder_name : string option val new_name : string -> ctxt -> string * ctxt val reserve_name : string -> ctxt -> ctxt end module Ctxt(R:Renaming) = struct include R let free_vars b = match b with | Box(_) -> empty_ctxt | Env(vs,_,_) -> let add ctxt (V(x)) = reserve_name (name_of x) ctxt in List.fold_left add empty_ctxt vs let new_var_in ctxt mkfree name = let (fresh_name, ctxt) = new_name name ctxt in (new_var mkfree fresh_name, ctxt) let new_mvar_in ctxt mkfree names = let ctxt = ref ctxt in let f name = let (v, new_ctxt) = new_var_in !ctxt mkfree name in ctxt := new_ctxt; v in (Array.map f names, !ctxt) let reset_ctxt closed ctxt = if reset_context_for_closed_terms && closed then empty_ctxt else ctxt let unbind_var constant mkfree name ctxt = match (constant, skip_constant_binders, constant_binder_name) with | (true, _ , Some(s)) -> (new_var mkfree s, ctxt) | (true, true, None ) -> (fst (new_var_in ctxt mkfree name), ctxt) | _ -> new_var_in ctxt mkfree name let unbind_in ctxt b = let ctxt = reset_ctxt (binder_closed b) ctxt in let (x, ctxt) = let constant = binder_constant b in unbind_var constant b.b_mkfree (binder_name b) ctxt in (x, subst b (b.b_mkfree x), ctxt) let unbind2_in ctxt b1 b2 = let ctxt = reset_ctxt (binder_closed b1 && binder_closed b2) ctxt in let (x, ctxt) = let constant = binder_constant b1 && binder_constant b2 in unbind_var constant b1.b_mkfree (binder_name b1) ctxt in let v = b1.b_mkfree x in (x, subst b1 v, subst b2 v, ctxt) let unmbind_in ctxt b = let ctxt = reset_ctxt (mbinder_closed b) ctxt in let ctxt_ref = ref ctxt in let xs = let fn i name = let constant = not b.mb_binds.(i) in let (x, ctxt) = unbind_var constant b.mb_mkfree name !ctxt_ref in ctxt_ref := ctxt; x in Array.mapi fn b.mb_names in (xs, msubst b (Array.map b.mb_mkfree xs), !ctxt_ref) let unmbind2_in ctxt b1 b2 = let ctxt = reset_ctxt (mbinder_closed b1 && mbinder_closed b2) ctxt in let ctxt_ref = ref ctxt in let xs = let fn i name = let constant = not b1.mb_binds.(i) && not b1.mb_binds.(i) in let (x, ctxt) = unbind_var constant b1.mb_mkfree name !ctxt_ref in ctxt_ref := ctxt; x in Array.mapi fn b1.mb_names in let vs = Array.map b1.mb_mkfree xs in (xs, msubst b1 vs, msubst b2 vs, !ctxt_ref) end (* Default renaming policy (functor instantiation) **************************) include Ctxt(struct (* Our renaming policy views variable names as a pair of a “prefix” and of a decimal “suffix”. For example, ["xyz42"] corresponds to [("xyz", 42)]. If a name does not have a decimal suffix, then the suffix value [0] is used. This means that ["xyz"] corresponds to [("xyz", 0)]. Moreover, all zeroes leading the decimal suffix are considered to be part of the prefix. Thus, name ["xyz0042"] corresponds to [("xyz00", 42)], and ["xyz0"] corresponds to [("xyz0", 0)]. Following the above convention, one way to represent contexts would be to have a map from prefixes to sets of suffixes. We could then check whether a name is in the context by checking whether its prefix is already mapped to a set containing its suffix. However, this is not what we do. Instead, we over-approximate the set of names contained in a context, and only store a map from prefixes to the largest used suffix. So, when there is a mapping from ["xyz"] to [42] in the context, this means that all the names of the form [("xyz", i)] are taken for [i <= 42]. *) type ctxt = int SMap.t let empty_ctxt = SMap.empty let reset_context_for_closed_terms = false let skip_constant_binders = false let constant_binder_name = None let split_name : string -> string * int = fun name -> let len = String.length name in (* [i] is the index of the first first character of the suffix. *) let i = let is_digit c = '0' <= c && c <= '9' in let first_digit = ref len in let first_non_0 = ref len in while !first_digit > 0 && is_digit name.[!first_digit - 1] do decr first_digit; if name.[!first_digit] <> '0' then first_non_0 := !first_digit done; !first_non_0 in if i = len then (name, 0) else (String.sub name 0 i, int_of_string (String.sub name i (len - i))) let get_suffix : string -> int -> ctxt -> int*ctxt = fun name suffix ctxt -> let n = try SMap.find name ctxt with Not_found -> -1 in let suffix = if suffix > n then suffix else n + 1 in (suffix, SMap.add name suffix ctxt) let merge_name : string -> int -> string = fun prefix suffix -> if suffix > 0 then prefix ^ string_of_int suffix else prefix let new_name : string -> ctxt -> string * ctxt = fun name ctxt -> let (prefix, suffix) = split_name name in let (suffix, ctxt) = get_suffix prefix suffix ctxt in (merge_name prefix suffix, ctxt) let reserve_name : string -> ctxt -> ctxt = fun name ctxt -> let (prefix, suffix) = split_name name in try let n = SMap.find prefix ctxt in if suffix <= n then ctxt else SMap.add prefix suffix ctxt with Not_found -> SMap.add prefix suffix ctxt end)