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) ->
@VictorTaelin
VictorTaelin / sat.md
Last active December 7, 2024 20:59
Simple SAT Solver via superpositions

Solving SAT via interaction net superpositions

I've recently been amazed, if not mind-blown, by how a very simple, "one-line" SAT solver on Interaction Nets can outperform brute-force by orders of magnitude by exploiting "superposed booleans" and optimal evaluation of λ-expressions. In this brief note, I'll provide some background for you to understand how this works, and then I'll present a simple code you can run in your own computer to observe and replicate this effect. Note this is a new observation, so I know little about how this algorithm behaves asymptotically, but I find it quite

@hirrolot
hirrolot / a-preface.md
Last active March 12, 2025 13:38
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 March 31, 2025 19:47
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
@VictorTaelin
VictorTaelin / itt-coc.ts
Last active January 26, 2025 18:02
ITT-Flavored Calculus of Constructions Type Checker
// A nano dependent type-checker featuring inductive types via self encodings.
// All computation rules are justified by interaction combinator semantics,
// resulting in major simplifications and improvements over old Kind-Core.
// Specifically, computable annotations (ANNs) and their counterpart (ANN
// binders) and a new self encoding based on equality (rather than dependent
// motives) greatly reduce code size. A more complete file, including
// superpositions (for optimal unification) is available on the
// Interaction-Type-Theory repository.
// Credits also to Franchu and T6 for insights.
@AndrasKovacs
AndrasKovacs / ZeroCostGC.md
Last active February 26, 2025 13:14
Garbage collection with zero-cost at non-GC time

Garbage collection with zero cost at non-GC time

Every once in a while I investigate low-level backend options for PL-s, although so far I haven't actually written any such backend for my projects. Recently I've been looking at precise garbage collection in popular backends, and I've been (like on previous occasions) annoyed by limitations and compromises.

I was compelled to think about a system which accommodates precise relocating GC as much as possible. In one extreme configuration, described in this note, there

type Fun<D, C> = Box<dyn Fn(D) -> C>;
trait Form {
type Repr<T>;
fn lam<A, B>(f: Fun<Self::Repr<A>, Self::Repr<B>>) -> Self::Repr<Fun<A, B>>;
fn appl<A, B>(f: Self::Repr<Fun<A, B>>, x: Self::Repr<A>) -> Self::Repr<B>;
}
@hirrolot
hirrolot / a-preface.md
Last active March 27, 2025 12:48
Higher-order polymorphic lambda calculus (Fω)
@hirrolot
hirrolot / cps-eval.ml
Last active March 27, 2025 12:49
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