Bruijn logo

bruijn

> A purely academic programming language based on lambda calculus and De > Bruijn indices written in Haskell. [Jump to examples](#Examples) or use the navigation tree to jump to other sections. Docs, articles, examples and more: [website](https://bruijn.marvinborner.de). ## Features - **De Bruijn indices[\[0\]](#References)** eliminate the complexity of α-equivalence and α-conversion - Unique **bracket-style representation** for lambda abstractions enables improved human-readability and faster syntactic perception - **Call-by-need** reduction with great time/memory complexity by using the RKNL[\[4\]](#References) abstract machine (similar to [calm](https://github.com/marvinborner/calm/)) - **Syntactic sugar** for unary/binary/ternary numerals and binary-encoded strings and chars - **No primitive functions** - every function is implemented in Bruijn itself - Highly space-efficient compilation to **binary lambda calculus (BLC)[\[2\]](#References)[\[3\]](#References)** additionally to normal interpretation and REPL - Strongly **opinionated parser** with strict syntax rules - **Recursion** can be implemented using combinators such as Y, Z or ω - Substantial **standard library** featuring many useful functions (see `std/`) ## Basics ### De Bruijn indices De Bruijn indices[\[0\]](#References) replace the concept of variables in lambda calculus. The index basically represents the abstraction layer you want to reference beginning at 0 with the innermost layer. For example, λx.x becomes λ0 because x referenced the first abstraction layer. Furthermore, λx.λy.xy becomes λλ10 and so forth. You can read more about De Bruijn indices on [Wikipedia](https://en.wikipedia.org/wiki/De_Bruijn_index). ### Syntax In general the syntax of bruijn is pretty similar to the previously presented normal lambda calculus syntax with De Bruijn indices. The main difference of the syntax of expressions is the usage of square brackets instead of λs (e.g. `[[1 0]]` instead of λλ10). You can use any function that you’ve previously defined. You can also overwrite previously defined functions. The environment gets interpreted from bottom to top (starting at `main`). The following are the main syntax specifications in the (minorly extended) [Backus-Naur form](https://en.wikipedia.org/wiki/Backus%E2%80%93Naur_form) (prefix/infix/mixfix operators are omitted for simplicity). ::= [a-ω,A-Ω,_][a-ω,A-Ω,0-9,?,!,',-]* ::= [A-Ω][a-ω,A-Ω]+ ::= "[" "]" ::= ("+" | "-")[0-9]+[u|b|t] ::= [0-9] ::= | | | "(" ")" | [namespace.] ::= ::= | ::= ":test " "(" ") (" ")" ::= ":import " [namespace] ::= ":input " ::= "# " * Differences in syntax between REPL and file: **For files**: The execution of a file begins at the `main` function. Its existence is mandatory. Note the missing equal sign in definitions. ::= ::= | | | | "\n" **For REPL**: ::= = ::= ":watch "