Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/ympbyc/Carrot

Purely functional lisp featuring: default currying, left-associative s-expression, lazy evaluation, static type system, and statically determined multimethods.
https://github.com/ympbyc/Carrot

Last synced: 2 months ago
JSON representation

Purely functional lisp featuring: default currying, left-associative s-expression, lazy evaluation, static type system, and statically determined multimethods.

Awesome Lists containing this project

README

        

Carrot
======

Dec 2012 Minori Yamashita

```lisp
;;./Carrot.scm examples/srfi-1.nadeko

(-> (Y (^ f (cons 1 (cons 1 (zipWith f + (cdr f))))))
-> (take 8)
-> reverse
<- (foldl (comp (comp ++ (++ " : ")) num->str) ""))
```

**Check out Carrot 2 (also WIP) too**

Installation
------------

+ Install gauche http://practical-scheme.net/gauche/index.html
+ Clone this repository

REPL
----

### To run the VM at the bleeding edge

```
./Carrot.scm examples/srfi-1.scm
```

The language
------------

This section specifies the language Carrot.
Carrot is a powerful functional programming language designed to be extremely simple and portable.

### Comments

Characters in a line after semicolon are treated as comments

**code 0** *comments*

```
;this is a comment
"this is not a comment" ;this is a comment
```

### Primitive values **(Expression)**

Primitive values are values that evaluates to itself. They include numbers, characters, strings, symbols and so on.
Implementations must provide at least the following primitives:

+ Strings
+ Numbers
+ Symbols

**code 1** *examples of primitive values*

```lisp
"abcd" ;string
1 ;number
'foo ;symbol
```

### Identifiers **(Expression)**

Identifiers are symbols that are or to be bound to another value. They are used to name functions, and as parameters of functions.
An identifier consists of one or more non-whitespace charactors. Some combinations are reserved as they are syntactic forms.

**code 2** *examples of identifiers*

```lisp
aaa
-he-lo-

@#=~
```

Implementations SHOULD allow every character that is not used for literals to be used to construct identifiers.

### Lambda expressions **(Expression)**

Lambda expressions create closures. Closures are objects that when applied to a value evaluate another value. They are basically, functions carrying its own environment with it.

Lambda expressions have the syntactic form
(^ *identifier ...* *expression*)
Where *identifier ...* is replaced with an arbitrary number of identifiers, and *expression* is replaced with an actual expression.
*identifier* is a parameter that gets bound to a value(actualy a thunk as we see later) when the closure is applied to arguments.

**code 3** *lambda expressions*

```lisp
(^ x (* x x)) ;;A

(^ a b c (* a (+ b c))) ;;B
```

The expression A, when applied, computes the squared value of a given number.
The expression B, when applied, performs a simple arithmetic operations on three numbers.
Note that the expression B is semantically the same as the following expression.

**code 4** *explicit currying*

```lisp
(^ a (^ b (^ c (* a (+ b c))))) ;;C
```

In fact, Carrot interprets the code B as C.
We will discuss the significance of it later.

### Definition (Statement)

The expression
(= (*name* *type* ...) *[identifier ...]* *expression*)
or
(=u (*name* *type* ...) *[identifier ...]* *expression*)
Binds the *expression* to the *name*.
If one or more parameters( *identifier ...* ) are given, they can be used in the *expression* to refer to the values the function is applied to. `=u` directs the compiler not to type-check the function body.

Multiple functions can share a name. In which case their types must differ.

**code 5** *defining a `map` function*

```lisp
(= (map (List a) (Fn a b) (List b))
xs f
(nil? xs nil
(cons (f (car xs))
(map (cdr xs) f))))
```

Implementations SHOULD implement tail call otimization.

### Application (Expression)

Functions(closures) can be applied to arguments. The application begins with an open paren, then the function, followed by a sequence of arguments, and ends with an corresponding close paren.

**code 6** *applying `map` function to a list*

```lisp
(map listX functionX) ;;A

(map (take integers 5) (+ 2)) ;;B
```

In the expression A, the map function defined in code 5 is applied to two arguments; listX and functionX assuming they are defined elsewhere.
This fills the parameter slots of the map function meaning that listX gets bound to xs and functionX gets bound to f. The expression inside map is then evaluated using both listX and functionX. Eventually the evaluation completes; producing a value(in this case a new list).

+ Carrot uses call-by-need evaluation strategy (although the implementation is incomplete).
+ Every function is curried.

Implementations SHOLD implement "call-by-need" instead of "call-by-name".

Type System
-----------

Carrot's type system is aimed to check if the programme's intention was consistent throughout the program. The type system is unique and somewhat criptic at first sight:

### How to Type Expressions

The syntactic form `=` gives types to identifiers.

```lisp
;; typing primitive values
(= (name String) "Rikka")

;; typing composit values
(= (names (List String)) (cons "Rikka" (cons "Kozue" (cons "Kamome" nil))))

;; typing functions
(= (second (List a) a) ;; ( ... )
xs ;; parameter...
(car (cdr xs))) ;; expression
```

### Twisted Algebraic Data Type

Carrot has no data structures except for closures, yet the type system is rich enough to express something like algebraic data types. Use `=u` for constructors instead of `=`.

```lisp
;; Lists
(=u (cons a (List a) (List a))
x xs f (f x xs))
(=u (nil (List a)) 'nil)
(= (car (List a) a)
xs (xs true))
(= (cdr (List a) (List a))
xs (xs false))
```

### Things Start to Get Odd

Carrot's runtime values are completely **untyped** (not even dynamicaly typed). As a consequence, Carrot's types are completely independent from its underlying implementation.

```lisp
;;invisible container

(=u (box a (Box a))
x x) ;; a boxed value's internal representation is the value itself
(=u (takeout (Box a) a)
x x) ;; just return what gets passed in

(box 7) ;;=> 7
(takeout 7) ;;=> TYPE ERROR!
(takeout (box 7)) ;;=> 7
```

Statically-Resolved MultiMethods
---------------------------------

Carrot supports multimethods. Because Carrot's runtime values are untyped, methods have to be selected at compile-time.

```lisp
(=u (dog Dog) :dog)
(=u (cat Cat) :cat)

(= (talk Dog String) _ "bow-wow")
(= (talk Cat String) _ "meoooow")

(talk cat)
```

Current implementation can't dispatch on generic types like `(List a)`.

S-expression without paren hell
-------------------------------

Carrot's default currying policy togather with lazyness gave us an unexpected gift -- Reduced use of parentheses.

Because the arguments are delayed automaticaly, we can implement booleans as functions.

```lisp
(= (true Bool) t e t)
(= (false Bool) t e e)

;scheme
(if (eq? a b) "equal" "not equal")

;carrot
(=? a b "equal" "not equal")
```

Pipeline operator in F#, and Synthread in Clojure are useful tool to avoid nesting of function calls. In Carrot, `->` can be used to compose functions left to right so it reads similarly to the synthread.

`->` is just a `(Fn b c) -> (Fn a b) -> c` function.

```lisp
(-> 1
-> (+ 2)
-> (* 3)
-> num->str
<- (flip ++ " = nine"))

;;is equivalent to
(++ (num->str (* (+ 1 2) 3)) " = nine")

(-> nil
-> (acons :name "立華")
-> (acons :favorite "ツナ")
-> (acons :job "ネットワークエンジニア")
<- (^ rikka (+++ (pull (assq :job rikka))
(pull (assq :name rikka))
(pull (assq :favorite rikka)))))
```

I/O
---

I/O in lazy languages are hard because the order of evaluation is not lexical, and if the return value of an I/O operation does not affect the overall result, the operation itself gets ommited. To go around this problem, Haskell uses what's called an IO Monad. IO Monad is a way to construct a computation, and sideeffects don't happen while the program is being evaluated.
Carrot takes a saner approach than that. It uses what we call a timed-io. Every sideeffectful function F takes an additional argument time X, and F that usually return `()` instead return a time Y. By giving Y to another sideeffectful function G, a clear relation between F and G gets formed so Carrot can figure out the evaluation order.
Every sideeffectful function may cache its results based on the time so it works fine even with call-by-name strategy.

```lisp
(-> (print 0 "Hi! What's your name?")
-> read
-> (^ name (print 1 (+++ "Nice to meet you, " name ". What do you like?")))
-> read
id (^ thing (print 2 (++ thing "? I like that, too!"))))
```

Macros
------

There is no macro mechanism built in to Carrot at this stage partly because there's littele need for it.

What's Coming or How to contribute
----------------------------------

Here are the list of features I intend to implement but haven't due to current dayjob situations (or because I'm flirting with Common Lisp all the time). I would love if someboday takes a shot at some of these. Be sure to email me (ympbyc at gmail) I'm lazy on github.

#### Call-by-Name Macro

Because Carrot is call-by-name, macro effect can be achieved with an operator which tells the compiler to suppress the evaluation of its operand. Type signature might be a good place to do this.

```scm
(= (example AST AST AST) x y
"We'll also need a way to construct lists more easily")
```

#### Proper Multimethod

Current implementation can't dispatch on composit datatypes. To make this work, some sort of templating mechanism is needed. You'll know what I mean if you try.

#### Lambda Folding and Constant Folding

Like Common Lisp compiler

#### Emacs SLIME support

Influenced by
--------------

Carrot is influenced by the following languages

+ Haskell - for lazyness and currying
+ Scheme - for syntax and actors
+ Io - for simplicity
+ Smalltalk-72 - for messaging notation
+ Clojure

License
-------

MIT

CHANGELOG
---------

### 1/22/2014

Statically-selected multimethod

### 1/4/2014

Integrated a type system.

### 11/17/2013

Krivine VM is replaced by S Machine described in http://www.cs.indiana.edu/cgi-bin/techreports/TRNNN.cgi?trnum=TR581

### 9/26/2013

A huge change in syntax.

+ `(:= (fname param ...) expr)` is replaced by `(= fname param ... expr)`
+ `(:= (name) const)` is replaced by `(= name const)`
+ `(-> (param ...) expr)` is replaced by `(^ param ... expr)`
+ `=` for equality check is replaced by `=?`

All the examples and tests are up to date with this change.