Carp/README.md

170 lines
6.9 KiB
Markdown
Raw Normal View History

2016-01-11 16:52:26 +03:00
# Carp
2016-01-11 16:55:03 +03:00
2016-01-18 20:15:58 +03:00
[![Join the chat at https://gitter.im/eriksvedang/Carp](https://badges.gitter.im/eriksvedang/Carp.svg)](https://gitter.im/eriksvedang/Carp?utm_source=badge&utm_medium=badge&utm_campaign=pr-badge&utm_content=badge)
2016-01-11 17:49:04 +03:00
<img src="https://github.com/eriksvedang/Carp/blob/master/img/temp_logo2.jpg" alt="Logo" align="right" />
2016-01-11 17:46:14 +03:00
2016-01-11 22:14:48 +03:00
<i>WARNING! This is a research project and a lot of information here might become outdated and misleading without any explanation. Don't try it out just yet!</i>
2016-01-11 17:57:24 +03:00
Carp is a small programming language designed to work well for interactive and performance sensitive use cases like games, sound synthesis and visualizations.
2016-01-11 17:46:14 +03:00
The key features of Carp are the following:
2016-01-11 18:59:43 +03:00
* Automatic and deterministic memory management (no garbage collector or VM)
2016-01-11 18:55:41 +03:00
* Inferred static types for great speed and reliability
2016-01-17 18:25:14 +03:00
* Live reloading of code, REPL-driven development, a fun and helpful workflow
2016-01-18 20:17:36 +03:00
* Ownership tracking enables a functional programming style while still using mutation of cache friendly data structures under the hood
2016-01-11 18:54:33 +03:00
* No hidden performance penalties allocation and copying is explicit
2016-01-11 18:36:31 +03:00
* Very good integration with exisiting C code
2016-01-11 17:57:24 +03:00
2016-01-19 15:39:21 +03:00
## A Small OpenGL/GLFW Example
2016-01-18 18:39:14 +03:00
```clojure
2016-01-19 15:39:21 +03:00
(defn gl-demo ()
2016-01-18 18:39:14 +03:00
(if (glfwInit)
(let [window (glfwCreateWindow 640 480 "Yeah!" NULL NULL)]
(if (null? window)
(panic "No window.")
2016-01-18 18:49:14 +03:00
(do (glfwMakeContextCurrent window)
2016-01-18 18:39:14 +03:00
(while (not (glfwWindowShouldClose window))
(do
(glClearColor 0.6f 0.85f 0.85f 1.0f)
(glClear gl-color-buffer-bit)
(glColor3f 1.0f 0.9f 0.2f)
2016-01-19 15:41:54 +03:00
(draw-rect -0.5f -0.5f 1.0f 1.0f)
2016-01-18 18:39:14 +03:00
(glfwSwapBuffers window)
(glfwPollEvents)))
(glfwTerminate))))
(panic "Failed to initialize glfw.")))
```
2016-01-19 15:32:41 +03:00
To build this example, load the gl bindings with ```(load-gl)```, then execute ```(bake-gl-demo-exe)``` to build an executable, or just ```(gl-demo)``` to run the program directly from the REPL.
2016-01-18 18:39:14 +03:00
## The Compiler
2016-01-18 19:16:09 +03:00
The Carp language is very tightly integrated with it's compiler which itself is written in a dynamic version of Carp (implemented in C). To work on a Carp program you run ```carp``` (first making sure it's in your $PATH, see installation instructions below) which starts the REPL. Everything you want to do to your program can be controlled from here.
2016-01-18 18:39:14 +03:00
For example, to compile a function named 'fib' you enter the following:
```clojure
λ> (bake fib)
```
2016-01-18 19:16:09 +03:00
This results in the compiler analyzing the code form for 'fib' and compiling it to (hopefully very fast) binary code, immediately loading this back into the REPL so that it can be called from there. The resulting C-code, ast and type signature are bound to the three variables 'c', 'ast' and 's', respectively. Inspecting their contents will teach you more about the innards of the Carp language, for sure!
2016-01-18 18:39:14 +03:00
2016-01-18 19:03:51 +03:00
From the REPL you can also inspect your the state of variables, extend the compiler, script the build process of your project, or statically analyze its code. All these operations should be really quick to execute and easy to remember so you can focus on developing your program.
2016-01-18 18:39:14 +03:00
2016-01-18 19:03:51 +03:00
To start the Carp compiler in development mode (which will run its test suite), invoke it like this instead:
2016-01-18 18:57:20 +03:00
2016-01-18 19:03:51 +03:00
```CARP_DEV=1 carp```
2016-01-18 18:57:20 +03:00
2016-01-18 18:39:14 +03:00
### Installation
2016-01-18 18:49:14 +03:00
2016-01-18 19:03:51 +03:00
Clone this repo, then run ```make``` in its root. Add the 'bin' directory to your path to enable calling the ```carp``` command. To do this, add the following to your .bashrc / .zshrc / whatever:
2016-01-13 16:12:42 +03:00
2016-01-14 14:04:34 +03:00
```export PATH=$PATH:~/Carp/bin/```
2016-01-13 16:12:42 +03:00
2016-01-19 15:38:05 +03:00
Carp is currently only tested on OSX 10.10. More platforms are coming soon. There are a few dependencies that have to be installed:
2016-01-18 18:49:14 +03:00
* libffi
* glfw3
* rlwrap
Note: 'rlwrap' is not strictly needed but makes the REPL experience much nicer, modify the '/bin/carp' script if you don't want to use it.
2016-01-18 18:39:14 +03:00
2016-01-18 18:57:20 +03:00
### Compiler Variables
* ```out-dir``` A string with the name of the folder where build artifacts should be put. Standard value is "".
* ```carp-dir``` The root folder of the Carp compiler, should be the same folder as this README.md file.
### Special Files
2016-01-18 19:07:32 +03:00
If a file called ```user.carp``` is placed in the folder ```~/.carp/```, that file will get loaded after the compiler has started. This file is meant for user specific settings that you want in all your projects, like little helper functions and other customizations.
2016-01-18 18:57:20 +03:00
2016-01-18 19:07:32 +03:00
If a file called ```project.carp``` is placed in the folder where you invoke the ```carp``` command this file will get loaded after the compiler has started (and after 'user.carp' has loaded). This files is intended for setting up the build process of this particular project, for example by loading the correct source files, configuring the compiler variables, etc.
2016-01-18 18:57:20 +03:00
2016-01-11 17:57:24 +03:00
## The Language
2016-01-11 18:36:31 +03:00
Carp borrows its looks from Clojure but the runtime semantics are much closer to those of ML or Rust. Here's a sample program:
2016-01-11 17:57:24 +03:00
```clojure
2016-01-11 18:36:31 +03:00
(defn fib (n)
(if (< n 2)
1
(+ (fib (- n 2)) (fib (- n 1)))))
2016-01-11 17:57:24 +03:00
```
2016-01-11 18:50:35 +03:00
This compiles to the equivalent of the following C program:
```C
int fib(int n) {
if(n < 2) {
return 1;
} else {
return fib(n - 2) + fib(n - 1);
}
}
```
2016-01-18 18:39:14 +03:00
The most important thing in Carp is to work with arrays of data. Here's an example of how that looks:
2016-01-11 19:07:19 +03:00
```clojure
2016-01-12 12:46:40 +03:00
(defn weird-sum (nums)
(reduce + 0 (map inc (filter even? nums))))
2016-01-11 19:07:19 +03:00
```
2016-01-18 19:09:13 +03:00
All the array modification functions like 'map', 'filter', etc. use C-style mutation of the array and return the same data structure back afterwards, no allocation or deallocation needed. The lifetime analyzer ("borrow checker" in Rust parlance) makes sure that the same data structure isn't used in several places.
2016-01-11 19:07:19 +03:00
2016-01-12 13:02:15 +03:00
### Data Literals
```clojure
100 ; int
3.14 ; float
2016-01-12 13:08:32 +03:00
true ; bool
2016-01-12 13:02:15 +03:00
"hello" ; string
2016-01-13 01:34:29 +03:00
'e' ; char
[1 2 3] ; array
2016-01-12 13:02:15 +03:00
```
### Special Forms
```clojure
(def variable-name value)
(defn function-name (arg1 arg2 ...) (function-body ...))
(let [var1 expr1, var2 expr2, ...] body)
(do expr1 expr2 ...)
(if expression true-branch false-branch)
(while expression body)
(for (i 0 100, j 0 100) body)
2016-01-19 13:21:04 +03:00
(ref x) ;; Turns an owned value into an unowned one
(reset! variable value)
2016-01-12 13:02:15 +03:00
```
2016-01-18 18:39:14 +03:00
### Structs (not implemented)
```clojure
(defstruct Vector2 [x :float, y :float])
(def my-pos (Vector2 102.2f 210.3f))
```
### Algebraic Data Types (not implemented)
2016-01-12 13:02:15 +03:00
```clojure
2016-01-12 13:13:09 +03:00
(defdata Color
RGB [r :float, g :float, b :float]
Grayscale [amount :float])
2016-01-12 13:02:15 +03:00
2016-01-12 13:13:09 +03:00
(def color (Grayscale 50.0f))
2016-01-12 13:02:15 +03:00
```
2016-01-12 13:13:09 +03:00
Omit the name tag to create a data constructor with the same name as the type:
2016-01-12 13:08:32 +03:00
```clojure
2016-01-12 13:13:09 +03:00
(defdata Vector3 [x :float, y :float, z :float])
2016-01-12 13:08:32 +03:00
2016-01-12 13:13:09 +03:00
(def position (Vector3 4.0 5.0 -2.0))
(def x-position (.x position)
2016-01-12 13:13:33 +03:00
```
2016-01-12 13:08:32 +03:00
2016-01-12 13:02:15 +03:00
### C interop
```clojure
(def blah (load-dylib "./libs/blah.so"))
(register blah "foo" (:int :int) :string) ;; will register the function 'foo' in the dynamic library 'blah' that takes two ints and returns a string
```
2016-01-11 19:07:19 +03:00
2016-01-19 15:29:06 +03:00
## License
This Source Code Form is subject to the terms of the Mozilla Public
License, v. 2.0. If a copy of the MPL was not distributed with this
file, You can obtain one at http://mozilla.org/MPL/2.0/.
2016-01-18 20:09:43 +03:00
© Erik Svedäng 2015 - 2016
2016-01-19 15:29:06 +03:00