/birb

:bird: Programming language that only consists of bird emojis

Primary LanguageHaskellMIT LicenseMIT

Birb

Birb is an advanced programming language that only consists of bird emojis 🐣. Each emoji gets substituted by a combinator bird of pure lambda calculus.

Birbs

Unfortunately, the Unicode standard does not yet have many (single-character) birds. These are the ones currently mapped/supported:

emoji animal combinator term
πŸ¦‰ owl owl $\lambda ab.b(ab)$
πŸ¦… eagle eagle $\lambda abcde.ab(cde)$
πŸͺ½ wing phoenix $\lambda abcd.a(bd)(cd)$
πŸ•ŠοΈ dove dove $\lambda abcd.ab(cd)$
🦜 parrot mockingbird $\lambda a.aa$
πŸ¦† duck quacky $\lambda abc.c(ba)$
🐀 touring chick turing $\lambda ab.b(aab)$
πŸ₯ kool chick kestrel $\lambda ab.a$
🐣 hatching chick quirky $\lambda abc.c(ab)$
🐦 simple bird identity $\lambda a.a$
🦚 peacock queer $\lambda abc.b(ac)$
🦀 dodo sage $\lambda ab.a(bb)\lambda ab.a(bb)$
🐧 penguin blackbird $\lambda abc.a(bc)$
🦒 swan substitution $\lambda abc.ac(bc)$
🦩 flamingo cardinal $\lambda abc.acb$

Lonely/unmatched birbs: πŸ”πŸ¦ƒπŸ“πŸͺΏ

Syntax

  • [birb]+: Birb
  • everything else: Comment

Syntax errors are impossible as long as you use at least one birb.

Semantics

Birbs stagger as they walk: they are reduced in alternating associative order, starting with left associativity at birb index $\lfloor\frac{\texttt{len}}{2}\rfloor$:

🐦🐦 -> (🐦🐦)
🐦🐦🐦 -> ((🐦🐦)🐦)
🐦🐦🐦🐦 -> (🐦((🐦🐦)🐦))
🐦🐦🐦🐦🐦 -> ((🐦((🐦🐦)🐦))🐦)
🐦🐦🐦🐦🐦🐦 -> (🐦((🐦((🐦🐦)🐦))🐦))
🐦🐦🐦🐦🐦🐦🐦 -> ((🐦((🐦((🐦🐦)🐦))🐦))🐦)
...

Examples

You can find more examples (with comments) in the samples/ directory.

Relationships

  • πŸͺ½πŸ¦ $\rightsquigarrow$ 🦒
  • 🦒🐦 $\rightsquigarrow$ πŸ¦‰
  • πŸ¦‰πŸ¦ $\rightsquigarrow$ 🦜
  • πŸ•ŠοΈπŸ¦ $\rightsquigarrow$ 🐧
  • 🐧🐧 $\rightsquigarrow$ πŸ•ŠοΈ
  • 🦩🐧 $\rightsquigarrow$ 🦚
  • 🦩🦚 $\rightsquigarrow$ 🐧
  • πŸ¦©πŸ¦† $\rightsquigarrow$ 🐣

One can only imagine what happens if two parrots talk to each other: 🦜🦜 $\rightsquigarrow$ πŸ’₯. The same happens with 🐀🐀; they just can’t stop waddling!

Arithmetic

For this example I use the Church numerals. Zero would then be encoded as πŸ₯🐦. The successor function can be written as 🦒🐧:

  • 🐦🐧🐦🦒🐧πŸ₯🐦 $\rightsquigarrow\lambda\lambda(10)$ – (Church numeral 1)
  • πŸ¦πŸ§πŸ¦πŸ§πŸ•ŠοΈπŸ¦’πŸ§πŸ¦’πŸ§πŸ₯🐦 $\rightsquigarrow\lambda\lambda(1(10))$ – (Church numeral 2)

Similarly, one can very obviously translate the Church addition function to πŸͺ½πŸ§. Now, to calculate $1+2$ based on their increments from zero:

  • πŸ¦πŸ¦πŸ•ŠοΈπŸ§πŸ•ŠοΈπŸ§πŸ¦πŸ§πŸ•ŠοΈπŸ§πŸ•ŠοΈπŸͺ½πŸ§πŸ¦’🐧🦒🐧πŸ₯🐦🦒🐧πŸ₯🐦 $\rightsquigarrow\lambda\lambda(1(1(10)))$ – (Church numeral 3)

Also: 🐧 is $a\cdot b$, 🦜 is $n^n$ and 🦚🐦 $a^b$.

Note that there exist many alternative ways to do arithmetic. Try writing the functions above with other birbs!

Containers

You can create a pair $\langle X,Y\rangle$ using 🦩🦩🦩YX.

Typically, one would now construct a list using repeated application of pairs (Boehm-Berarducci/Church encoding). However, due to the reversed application and alternating associativity, the Mogensen-Scott encoding is more suitable:

List $\langle X_1,X_2,\dots,X_n\rangle$: [🦩]ⁿ🦩X2X1...XN.

Busy beavers birbs

Contestants:

  • The touring eagle: [🐦]ⁿ[πŸ¦…πŸ€]ⁿ ($n=3$: 9 birbs, ~20M BLC bits)
  • better? PR!

Transpiler

I created a lambda calculus to Birb transpiler. It works by converting binary lambda calculus to SKI combinators, which then get converted to Jot and back to SKI combinators. The resulting SKI combinators then get converted to Birbs.

The reason I convert to Jot first is that Birbs semantics don’t allow trivial transpilation from arbitrary SKI expressions. With Jot however, applications with non-associative expressions like ((s k) (k s)) are impossible, as only a single non-associative application exists (the innermost/deepest expression).

With all these conversions, the resulting transpiled Birb code is big, ugly and slow. There should be a way to optimize the transpilation process, but it’s probably a bit more complicated.

Usage

Install Haskell’s stack. Then,

  • stack run -- reduce file.birb or stack run -- reduce <(echo 🐧🐧)
  • stack run -- transpile <(echo 01000110100000011100111001110011100111010) to generate a birb program that calculates $5^5$
  • stack install so you can enjoy birb from anywhere

If the output cannot be translated to birbs, the raw lambda calculus term (with De Bruijn indices) is printed. For the examples above I sometimes manually converted the term back to birbs.

Turing-completeness

Birb is Turing complete, since one can construct any term of the Jot variant of Iota. A Jot term ((X s) k) is equivalent to 🐦🐦X🦒πŸ₯. Similarly, (s (k X)) is equivalent to πŸ¦πŸ¦†XπŸ₯🦒.


The idea of the language was originally proposed in 2021 by @SCKelement on Esolang.