• Stars
    star
    134
  • Rank 270,872 (Top 6 %)
  • Language
    C
  • License
    MIT License
  • Created over 2 years ago
  • Updated 2 months ago

Reviews

There are no reviews yet. Be the first to send feedback to the community and the maintainers!

Repository Details

build-once run-anywhere OCaml programs

Esperanto

Build-once Run-anywhere OCaml programs


Esperanto makes OCaml a build-once run-anywhere language, like itself, except it doesn't need an interpreter or virtual machine. Indeed, OCaml is able to produce a native executable which requires few libraries:

  • a standard C library
  • libasmrun.a (the caml-runtime)
  • probably libunix.a if you depend on unix.cmxa

Esperanto replaces the host's C library by Cosmopolitan. Then, it outputs a POSIX-approved polyglot format that runs on many platforms. For more details, please read the αcτµαlly pδrταblε εxεcµταblε.

Getting Started

You can install the project with OPAM:

$ opam pin add -y https://github.com/dinosaure/esperanto.git

A simple program

Let's start to make a simple Hello World! example:

$ cat >main.ml <<EOF
let () = print_endline "Hello World!"
EOF
$ ocamlfind -toolchain esperanto opt main.ml
$ objcopy -S -O binary a.out
$ file a.out
a.out: DOS/MBR boot sector
$ sh -c "./a.out"
Hello World!

With dune and dependencies

You can take your favorite project and compile executables with the Esperanto toolchain. You need to specify a new dune-workspace:

$ git clone https://github.com/dinosaure/hxd.git
$ cd hxd
$ cat >>bin/dune <<EOF

; A rule to properly strip everythings
(rule
 (target xxd.com)
 (enabled_if
  (= %{context_name} esperanto))
 (mode promote)
 (deps xxd.exe)
 (action (run objcopy -S -O binary %{deps} %{target})))
EOF
$ cat >dune-project<<EOF
(lang dune 2.0)

(context (default))

(context
 (default
  (name esperanto)
  (toolchain esperanto)
  (merlin)
  (host default)))
EOF

Finally, as a static program, you must fetch dependencies with opam monorepo and build your project with them:

$ opam monorepo lock
$ opam monorepo pull
$ dune build bin/xxd.com
$ ls bin/xxd.com

Now, you have a portable/polyglot program. The user can assimilate the program with the --assimilate option:

$ file bin/xxd.com
bin/xxd.com: DOS/MBR boot sector
$ sh -c "./bin/xxd.com --assimilate"
$ file bin/xxd.com
bin/xxd.com: ELF 64-bit LSB executable

The --assimilate option modify the executable itself to become really native to your platform - in other words, the executable no longer becomes polyglot!

The toolchain

The cosmopolitan toolchain (see toolchain/) provides a special compiler and a special ld script which integrates options needed to well-build the caml runtime with Cosmopolitan.

It puts into your OPAM 3 binaries:

  • arch-esperanto-none-static-cc the compiler
  • arch-esperanto-none-static-ld the linker
  • arch-esperanto-none-static-objinfo a simple tool to mainly strip binaries

It installs few objects files:

  • cosmopolitan.a the Cosmopolitan library
  • crt.o, ape.o, ape-no-modify-self.o and ape.lds
  • startup.o which is a well-builded caml_startup for your binaries which must be linked with -z caml-startup option
  • startup_unix.o/fake_unix.o which initiates few constants needed by libunix.a if you depend on unix.cmxa

The caml compiler

From the toolchain, we are able to compile the OCaml compiler and its runtime with Cosmopolitan (with few fixes). esperanto provides a new OCaml toolchain which can be used by dune to "cross"-compile a project.

Such design comes from MirageOS and Solo5. For C stubs, we provide the __ESPERANTO__ definition which permits to orchestrate your compilation of C files according to the OCaml toolchain.

Currently, we support OCaml 4.13 & 4.14.

Issues and bugs

zsh and binfmt_misc

Currently, zsh < 5.9.0 does not support well Cosmopolitan/APE binaries. However, the recent version fix the initial issue. In the case of you have zsh < 5.9.0, you can use the --assimilate option to modify the executable to a real native application:

$ zsh --version
zsh 5.8.1
$ zsh
$ ./a.out
zsh: exec format error: ./a.out
$ sh -c "./a.out --assimilate"
$ ./a.out
Hello World!

However, as explained above, the binary no longer becomes polyglot!

An other issue on Linux is about binfmt_misc which wants to interpret the binary as a Windows executable (due to the header of the Cosmopolitan binary). A good explanation is available here: jart/cosmopolitan#2

The Cosmopolitan distribution (and specially the APE distribution) gives an installer: https://justine.lol/apeloader/#binfmt_misc This installer learns to binfmt_misc how to recognize and execute a Cosmopolitan binary.

pthread and platforms

Cosmopolitan provides partially pthread (only mutex) and few functions which can simulate threads (see spawn and join). Due to the partial implementation of the pthread, we are not able to provide threads.cmxa from the OCaml project. However, upstream wants to improve this situation and, in the near future, Cosmopolitan will probably provide a full implementation of pthread. In this future, we will be able to provide threads.cmxa and start the support of OCaml 5.0.

Warnings and C stubs

The hard part of esperanto is when we need to compile C stubs with the Cosmopolitan C library. Indeed, the context is a bit different and your C code probably does not compile with Cosmopolitan. The usual mistake is about constants:

static int wait_flag_table[] = { WNOHANG, WUNTRACED };

value wait4(value flags, ...) {
  int flags = caml_convert_flag_list(flags, wait_flag_table);
  ...
}
type wait_flag = WNOHANG | WUNTRACED
external wait4 : wait_flag list -> ... = "wait4"

Such code can not compile with Cosmopolitan due to the fact that WNOHANG & WUNTRACED are not constants but values which are set at the beginning of your program according to your platform.

Some others issues can remain but they are probably specific to you code. But Esperanto can not fit for every projects, so third-party libraries maintainers should be aware about "these details" if they want to support Esperanto.

More Repositories

1

bob

A peer-to-peer file-transfer tool in OCaml
OCaml
134
star
2

pasteur

Paste-eur as unikernel
OCaml
65
star
3

gilbraltar

MirageOS on RPi4
C
58
star
4

art

Adaptive Radix Tree in OCaml
OCaml
48
star
5

paf-le-chien

Port of HTTP/AF & H2 with Mirage and mimic
OCaml
31
star
6

hxd

J'ai plus mal à la tête maintenant
OCaml
27
star
7

docteur

An opiniated file-system for MirageOS
OCaml
25
star
8

multipart_form

According an other RFC2388...
OCaml
25
star
9

cri

IRC protocol in OCaml
OCaml
25
star
10

carton

Implementation of the PACK file (Git) in OCaml
OCaml
21
star
11

mimic

A dynamic way to instantiate an OCaml flow
OCaml
18
star
12

tuyau

Should be the next conduit ... or not
OCaml
17
star
13

spoke

SPAKE2+EE implementation in OCaml
C
16
star
14

facteur

Send a mail in OCaml
OCaml
15
star
15

blaze

OCaml
15
star
16

dsh

A functionnal programming language
OCaml
15
star
17

contruno

A TLS termination proxy as a MirageOS
OCaml
13
star
18

minifiber

Parallel computation in OCaml - multicore for really poor people
OCaml
12
star
19

dinoscheme

Un langage de programmation inspiré de Scheme et d'OCaml (Compilateur + Machine Virtuelle)
OCaml
11
star
20

sirodepac

Pack file encoder/decoder in OCaml
OCaml
11
star
21

tarides-workshop

C
8
star
22

demilekarantuite

2048 in OCaml
OCaml
8
star
23

z

DON'T USE IT
C
7
star
24

buffet

OCaml
7
star
25

sage

Sage and Agnostic I/O
OCaml
7
star
26

usine

Benchmarking tool for OcsigenServer and other in OCaml
OCaml
6
star
27

radis

« Un énorme radis. »
OCaml
6
star
28

ocaml-maildir

OCaml
6
star
29

ocaml-dmarc

OCaml
5
star
30

blogger

OCaml
5
star
31

rainbow

Red-black tree implementation in C with OCaml
OCaml
5
star
32

overlap

Bigarray.overlap as a library
OCaml
5
star
33

prettym

A simple bounded encoder constraints by columns in OCaml
OCaml
5
star
34

unstrctrd

OCaml
4
star
35

deadaf

Shell
4
star
36

caravan

Fixup BSS and inject new section into ELF binary
OCaml
4
star
37

ehouais

EWAH in OCaml (PoC)
HTML
3
star
38

gas-eqaf

Abstract interpretation of GAS assembler for eqaf project
OCaml
3
star
39

rev-list

git rev-list in OCaml
OCaml
3
star
40

mirage-os-shinimy

OCaml
2
star
41

dcpr-inv

Inverted stubs of decompress (OCaml to C)
Makefile
2
star
42

uuuui

Universal Unifier to Unicode Un OCaml (Importation tool)
OCaml
2
star
43

ada-language-19-05-2023

OCaml
2
star
44

not-so-smart

Implementation of the Git protocol
OCaml
2
star
45

conduit-dev

2
star
46

chamonix

Chamo, Chamoni, Chamonix
OCaml
1
star
47

ogaml

Jeu vidéo en OCaml et ocamlsdl
OCaml
1
star
48

lavoisier

lol, you should be lost ...
OCaml
1
star
49

mirage-dune

The smallest opam repository to have MirageOS with dune
Shell
1
star
50

short-hash

HTML
1
star
51

mirage.irc.libera.chat

logs on #[email protected]
1
star
52

-

Coq in my dream
Coq
1
star
53

din.osau.re

Site développé avec les technologies Ruby
Ruby
1
star
54

blog.osau.re

Le blog d'un Dinosaure
CSS
1
star
55

tuyau-example

A simple example to download an HTML page and be free to use TLS or not when it's available.
OCaml
1
star
56

mirage-entropy-nocrypto

Functorized entropy engine for ocaml-nocrypto
OCaml
1
star
57

blog.x25519.net

HTML
1
star
58

ULC

Untyped Lambda-Calculus in OCaml
OCaml
1
star
59

mirage-nolink

Which library I need to compile my unikernel? Where is nocrypto's Native module?
OCaml
1
star