Skip to content

Instantly share code, notes, and snippets.

@hirrolot
hirrolot / a-classic.ml
Last active December 1, 2024 21:28
Various homeomorphic embedding implementations
type term = Call of string * term list
let rec decide_he (s, (Call _ as t)) =
he_by_diving (s, t) || he_by_coupling (s, t)
and he_by_diving (s, Call (_h', args')) =
List.exists (fun t -> decide_he (s, t)) args'
and he_by_coupling (Call (h, args), Call (h', args')) =
if h = h' then List.for_all2 (fun s t -> decide_he (s, t)) args args'
@hirrolot
hirrolot / transpose-args.ml
Last active October 1, 2024 16:47
Nondeterministic substitution by matrix transposition
type term = Var of string | Call of string * term array
[@@deriving show { with_path = false }]
type possibilities = term array [@@deriving show]
let demux ~(x : string) ~(values : term array) : term -> possibilities =
let rec go = function
| Var y when x = y -> values
| Var _ as t -> Array.map (fun _ -> t) values
| Call (op, args) ->
@hirrolot
hirrolot / a-preface.md
Last active December 19, 2024 09:32
A complete implementation of the positive supercompiler from "A Roadmap to Metacomputation by Supercompilation" by Gluck & Sorensen

This is the predecessor of Mazeppa.

Supercompilation is a deep program transformation technique due to V. F. Turchin, a prominent computer scientist, cybernetician, physicist, and Soviet dissident. He described the concept as follows [^supercompiler-concept]:

A supercompiler is a program transformer of a certain type. The usual way of thinking about program transformation is in terms of some set of rules which preserve the functional meaning of the program, and a step-by-step application of these rules to the initial program. ... The concept of a supercompiler is a product of cybernetic thinking. A program is seen as a machine. To make sense of it, one must observe its operation. So a supercompiler does not transform the program by steps; it controls and observes (SUPERvises) the running of the machine that is represented by th

@hirrolot
hirrolot / terminating-NbE.ml
Last active October 1, 2024 16:44
Terminating untyped NbE with a configurable limit
module Make_term (S : sig
type 'a t [@@deriving show]
end) =
struct
type t = def S.t
and def = Lam of t | Var of int | Appl of t * t
[@@deriving show { with_path = false }]
let _ = S.show
@hirrolot
hirrolot / a-preface.md
Last active October 1, 2024 16:56
Higher-order polymorphic lambda calculus (Fω)
@hirrolot
hirrolot / cps-eval.ml
Last active October 1, 2024 16:45
A simple CPS evaluation as in "Compiling with Continuations", Andrew W. Appel
type cps_var =
(* Taken from the lambda term during CPS conversion. *)
| CLamVar of string
(* Generated uniquely during CPS conversion. *)
| CGenVar of int
type cps_term =
| CFix of (cps_var * cps_var list * cps_term) list * cps_term
| CAppl of cps_var * cps_var list
| CRecord of cps_var list * binder
@hirrolot
hirrolot / a-main.ml
Last active October 1, 2024 16:57
A simple CPS conversion as in "Compiling with Continuations", Andrew W. Appel
(* A variable identifier of the lambda language [term]. *)
type var = string [@@deriving eq]
(* The lambda language; direct style. *)
type term =
| Var of var
| Fix of (var * var list * term) list * term
| Appl of term * term list
| Record of term list
| Select of term * int
@hirrolot
hirrolot / CoC.ml
Last active December 21, 2024 23:52
How to implement dependent types in 80 lines of code
type term =
| Lam of (term -> term)
| Pi of term * (term -> term)
| Appl of term * term
| Ann of term * term
| FreeVar of int
| Star
| Box
let unfurl lvl f = f (FreeVar lvl)
@hirrolot
hirrolot / untyped-hoas.ml
Last active October 1, 2024 17:08
Untyped lambda calculus with HOAS and De Bruijn levels
type term = Lam of (term -> term) | Appl of term * term | FreeVar of int
let unfurl lvl f = f (FreeVar lvl)
let unfurl2 lvl (f, g) = (unfurl lvl f, unfurl lvl g)
let rec print lvl =
let plunge f = print (lvl + 1) (unfurl lvl f) in
function
| Lam f -> "(λ" ^ plunge f ^ ")"
| Appl (m, n) -> "(" ^ print lvl m ^ " " ^ print lvl n ^ ")"
@hirrolot
hirrolot / stlc-form-meaning-de-bruijn.ml
Last active December 18, 2024 02:10
A simply typed lambda calculus in tagless-final (De Bruijn indices)
module type Form = sig
type ('env, 'a) meaning
val vz : ('a * 'env, 'a) meaning
val vs : ('env, 'a) meaning -> ('b * 'env, 'a) meaning
val lam : ('a * 'env, 'b) meaning -> ('env, 'a -> 'b) meaning
val appl :
('env, 'a -> 'b) meaning -> ('env, 'a) meaning -> ('env, 'b) meaning
end