Skip to content

meric/l2l

Repository files navigation

l2l

This language is a superset of Lisp and Lua.

Example

boolean.lisp, implements and, or special forms:

@import quasiquote
@import quote
@import fn
@import local
@import do
@import let
@import cond

(fn circuit_and (invariant cdr output truth)
  (cond cdr
    (let (
      car (:car cdr)
      ref (lua_name:unique "_and_value"))
      `\
        local \,ref = \,\(expize(invariant, car, output))
        if \,ref then
          \,(cond (:cdr cdr)
              (circuit_and invariant (:cdr cdr) output truth)
              `\\,truth = \,ref)
        else
          \,truth = false
        end)))

(fn expize_and (invariant cdr output)
  (let (ref (lua_name:unique "_and_bool"))
    (table.insert output `\local \,ref = true)
    (table.insert output (circuit_and invariant cdr output ref))
    ref))

(fn statize_and (invariant cdr output)
  (to_stat (expize_and invariant cdr output)))

(fn circuit_or (invariant cdr output truth)
  (cond cdr
    (let (
      car (:car cdr)
      ref (lua_name:unique "_or_value"))
      `\
        if not \,truth then
          local \,ref = \,\(expize(invariant, car, output))
          if \,ref then
            \,truth = \,ref
          end
        end
        \,(cond (:cdr cdr)
            (circuit_or invariant (:cdr cdr) output truth)))))

(fn expize_or (invariant cdr output)
  (let (ref (lua_name:unique "_or_bool"))
    (table.insert output `\local \,ref = false)
    (table.insert output (circuit_or invariant cdr output ref))
    ref))

(fn statize_or (invariant cdr output)
  (to_stat (expize_or invariant cdr output)))

{
  lua = {
    ["and"] = {expize=expize_and, statize=statize_and},
    ["or"] = {expize=expize_or, statize=statize_or}
  }
}

Quick Start

# Requires https://github.com/meric/leftry cloned as a sibling to this repo.
git clone [email protected]:meric/leftry.git
git clone [email protected]:meric/l2l.git
cd l2l
make clean
make test
make repl

Run the following commands:

> (print "hello world")
hello world
> \print("hello world")
hello world
> (let (a 1) (print `\print(\,a)))
print(1)
> (let (a 1) (print (getmetatable `\print(\,a))))
lua_block

Syntax Highlighting

l2l-syntax-highlighting

Naming

Lisp names are mangled into Lua by replacing non lua compliant characters with lua compliant characters. (See the mangle function in l2l/reader.lua).

Lisp names can contain dashes, dots, alphabets, numbers, underscores, and many more characters, but they must not consist of two dots consecutively unless the name is .. (lua string concat) or ... (lua vararg).

This is so the compiler can mangle lua field accessor names my_table.my_subtable-with-dashes.some_key properly.

Features

  • Mix Lisp and Lua in source code with backslash.

    \print(\(+ 1 2 3 4))
  • Quasiquoting Lua expressions.

    (table.insert output `\local \,ref = false)
  • Macro and special form aliasing.

  • Macro as modules.

    @import (let x); (x.let (y 1) (print y))
  • Custom special forms as modules.

    For example, boolean.lisp.

  • Zero-cost map, filter, reduce abstractions.

  • Implement special forms that can inline anonymous functions as macros.

  • Special forms in Lua.

    @import iterator
    \
    map(function(x) return x + 2 end,
      filter(function(x) return x % 2 == 0 end,
        map(function(x) return x + 1 end, {1, 2, 3, 4})))

    Compiles into (nested loops collapsed into a single pass):

    local ipairs = require("l2l.iterator")
    local vector = require("l2l.vector")
    local next38,invariant37,i39 = ipairs({1,2,3,4});
    local values41 = vector();
    while i39 do
      local v40;i39,v40=next38(invariant37,i39);
      if i39 then
        v40=v40 + 1;
        if v40 % 2 == 0 then
          v40=v40 + 2;
          (values41):insert(v40)
        end
      end
    end
    return values41

Philosophy

The Tao begot one. One begot two.
Laozi

Now I do not know whether I was then a man dreaming I was a butterfly,
Or whether I am now a butterfly, dreaming I am a man.
Zhuangzi

I have put duality away, I have seen that the two worlds are one;
One I seek, One I know, One I see, One I call.
Jalaluddin Rumi

Now I do not know whether I was writing Lua inside of Lisp,
Or whether I am now writing Lisp, inside of Lua.
I have put duality away, I have seen that the two worlds are one;
One I read. One I write. One I compile. One I run.
You

Obligatory

I am the servant of the Qur'an as long as I have life.
I am the dust on the path of Muhammad, the Chosen one.
If anyone quotes anything except this from my sayings,
I am quit of him and outraged by these words.
Jalaluddin Rumi