SML, the revenge

I love SML. After some evenings and nights with Java, I’m really glad to go back to this old but still good language. Some of my work in progress can be find there: TheReturnOfTheSonOfSML. In this post, I just recall some basic rules that I forget after a few weeks of inactivity in this domain.

interpreters and compilers

The easiest implementation is PolyML which have an interpreter and a good compiler. The project is still alive, which is very important.

MLton is a good compiler, which is easy to use but without interpreter. The last version is from February of this year and the previous from 2013. Low activity but not null.

The historic SML/NJ implementation offers an interpreter with a compiler. The latter is not easy to use and I won’t.

There are more but today I will only use these.

a sample program

There is a great hello project which can help us to learn how to structure a source for PolyML and MLTon. If there is a strong standardisation of the language, it’s not the case of the compiling system. It’s why we must do different things for each implementation.

PolyML, first steps

It’ a very permissive system whith one rule: you must provide a main function as the entry point of your program:

fun main() =
	print "Hello, world!\n"

Put that code in a file called hello1.sml and run this:

$ polyc -o hello1.exe hello1.sml
$ ./hello1.exe
Hello, world!

MLTon, first steps

Now, testing with MLTon give a unexpected result:

$ mlton hello1.sml
$ ./hello1

If we create a main.sml like this:

(* main.sml *)

val _ = main ()

and a sort of project file hello1.mlb like that:

$(SML_LIB)/basis/basis.mlb

hello1.sml
main.sml

The test can give us now what we expect:

$ mlton -output hello1.mlt.exe -verbose 1 hello1.mlb
MLton 20180207
MLton starting
   Compile SML starting
      pre codegen starting
      pre codegen finished in 2.28 + 0.71 (24% GC)
      amd64 code gen starting
      amd64 code gen finished in 0.15 + 0.00 (0% GC)
   Compile SML finished in 2.42 + 0.71 (23% GC)
   Compile and Assemble starting
      gcc -c -fPIC -DPIC -I/usr/local/lib/mlton/targets/self/include \
          -std=gnu11 -fno-common -O1 -fomit-frame-pointer \
          -fno-strict-aliasing -w -I/usr/local/lib/mlton/include -m64 -o \
          /tmp/filetXRrV4.o /tmp/filev8cCPy.1.c
      gcc -c -m64 -o /tmp/fileNPOJQ5.o /tmp/filedMrdjS.0.s
   Compile and Assemble finished in 0.14 + 0.00 (0% GC)
   Link starting
      gcc -o hello1.mlt.exe /tmp/filetXRrV4.o /tmp/fileNPOJQ5.o \
          -L/usr/local/lib/mlton/targets/self -lmlton-pic -lgdtoa-pic -lm \
          -lgmp -m64 -Wl,-znoexecstack
   Link finished in 0.03 + 0.00 (0% GC)
MLton finished in 2.64 + 0.72 (21% GC)
$  ./hello1.mlt.exe
Hello, world!

The -verbose option give some indication of the evolution of the compilation. Even if it’s verbose, it’s useful for big project because the compiler is really slower than polyc.