Skip to content

Latest commit

 

History

History
162 lines (134 loc) · 4.96 KB

readme.md

File metadata and controls

162 lines (134 loc) · 4.96 KB

Lisp Simulacrum

Description

This is a toy Lisp interpreter implemented in Python3. I wanted to learn more about how programming languages work and decided to make a mostly functional Lisp. The tokenizer and parser are based on reading Dragon Book with the lexer being an implementation of a FSA and the parser being a basic recursive-descent parser. The evaluator was almost completely ripped from SICP, with a nice tweak from Peter Norvig's (An ((Even Better) Lisp) Interpreter (in Python)), and a couple of my own ideas on application evaluation, in particular, for apply and macros. Having said that, any bugs are entirely due to me.

This is nowhere near a production quality interpreter, but it's not too hard to understand and might be useful as a learning tool. The style is not always super Pythonic, and there are many places where I chose an approach closer to how I would implement in a lower level language. To run, simply run reader.py, I recommend using rlwrap to make navigation easier.

Syntax and Not-So-Niceness

The syntax is some combination of Scheme and Common Lisp. For example, basic function definition look like:

(define (square x)
  (* x x))

More complicated functions, however, borrow some syntax from Common Lisp:

(define (foo x &optional y)
  (if (null? y)
    x
    y))

and

(define (foo x &rest rest)
  (if (null? rest)
    x
    rest))

For optional arguments, the default is nil, someday I'd like to add user-supplies defaults.

Macros are also of a more Common Lisp flavor.

(define-macro (plus x y)
  `(+ ,x ,y))

Like Common Lisp, macros are not hygeinic, which means variable capture is a problem.

>>(define-macro (swap x y)
    `(let ((value ,x))
       (set! ,x ,y)
       (set! ,y ,value)))
swap
>>(define x 1)
x
>>(define value 2)
y
>>(swap value x)
2
>>value
2
>>x
2

A basic gensym function has been added to help prevent this.

>>(define-macro (swap x y)
    (let ((value (gensym)))
      `(let ((,value ,x))
         (set! ,x ,y)
         (set! ,y ,value))))
swap
>>(define x 1)
1
>>(define value 2)
2
>>(swap value x)
2
>>value
1
>>x
2

What Works?

Tail Call Optimization

Quite a bit works out of the box. The main eval loop is done in a way that we get some tail-call optimization, thus we can do things like

>>(define (fib-iter n a b)
    (if (= n 0)
      a
      (fib-iter (- n 1) b (+ a b))))
fib-iter
>>(define (fib n)
    (fib-iter n 0 1))
fib
>>(fib 10000)
3364476...6875

without smashing Python's stack. The downside to the implementation is that it's pretty slow, even moreso that it's being implemented in a high-level language like Python.

Macros

As mentioned above, Lisp Simulacrum has basic macro functionality.

>>(define-macro (plus coll)
    `(+ ,@coll))
plus
>>(plus (1 2 3))
6

A Good Amount of Builtins

builtins.py has premade definitions for several useful functions, e.g. c*r of length 3, length function for lists, list-ref, map, and reduce.

Want to know more about Lisp?

That's great! Here are some wonderful places to start, in no particular order.

SICP

On Lisp

Practical Common Lisp

Let Over Lambda

Land of Lisp

Want to know more about language implementation?

There are a ton of books out there, and everyone seems to like precisely one and hate all the rest. My path was to read SICP up through chapter 4 (metalinguistic abstraction) where they build a Lisp in Scheme. Then I picked up a copy of Dragon Book and read up to somewhere in Chapter 6. Some other books I've had recommended to me are

Modern Compiler Implementation in C

Language Implementation Patterns

Programming Language Pragmatics

And, while not a books, I whole heartedly recommend reading Peter Norvig's articles

(How to Write a (Lisp) Interpreter (in Python))

(An ((Even Better) Lisp) Interpreter (in Python))

As well as

pylisp

psyche

pyscheme

And if you're interested in some lower level Schemes

Scheme 9 from Empty Space

TinyScheme