123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427 |
- ================================
- Nim Backend Integration
- ================================
- :Author: Puppet Master
- :Version: |nimversion|
- .. contents::
- "Heresy grows from idleness." -- Unknown.
- Introduction
- ============
- The `Nim Compiler User Guide <nimc.html>`_ documents the typical
- compiler invocation, using the ``compile`` or ``c`` command to transform a
- ``.nim`` file into one or more ``.c`` files which are then compiled with the
- platform's C compiler into a static binary. However, there are other commands
- to compile to C++, Objective-C, or JavaScript. This document tries to
- concentrate in a single place all the backend and interfacing options.
- The Nim compiler supports mainly two backend families: the C, C++ and
- Objective-C targets and the JavaScript target. `The C like targets
- <#backends-the-c-like-targets>`_ creates source files that can be compiled
- into a library or a final executable. `The JavaScript target
- <#backends-the-javascript-target>`_ can generate a ``.js`` file which you
- reference from an HTML file or create a `standalone Node.js program
- <http://nodejs.org>`_.
- On top of generating libraries or standalone applications, Nim offers
- bidirectional interfacing with the backend targets through generic and
- specific pragmas.
- Backends
- ========
- The C like targets
- ------------------
- The commands to compile to either C, C++ or Objective-C are:
- //compileToC, cc compile project with C code generator
- //compileToCpp, cpp compile project to C++ code
- //compileToOC, objc compile project to Objective C code
- The most significant difference between these commands is that if you look
- into the ``nimcache`` directory you will find ``.c``, ``.cpp`` or ``.m``
- files, other than that all of them will produce a native binary for your
- project. This allows you to take the generated code and place it directly
- into a project using any of these languages. Here are some typical command-
- line invocations::
- $ nim c hallo.nim
- $ nim cpp hallo.nim
- $ nim objc hallo.nim
- The compiler commands select the target backend, but if needed you can
- `specify additional switches for cross-compilation
- <nimc.html#crossminuscompilation>`_ to select the target CPU, operative system
- or compiler/linker commands.
- The JavaScript target
- ---------------------
- Nim can also generate `JavaScript`:idx: code through the ``js`` command.
- Nim targets JavaScript 1.5 which is supported by any widely used browser.
- Since JavaScript does not have a portable means to include another module,
- Nim just generates a long ``.js`` file.
- Features or modules that the JavaScript platform does not support are not
- available. This includes:
- * manual memory management (``alloc``, etc.)
- * casting and other unsafe operations (``cast`` operator, ``zeroMem``, etc.)
- * file management
- * OS-specific operations
- * threading, coroutines
- * some modules of the standard library
- * proper 64-bit integer arithmetic
- To compensate, the standard library has modules `catered to the JS backend
- <lib.html#pure-libraries-modules-for-js-backend>`_
- and more support will come in the future (for instance, Node.js bindings
- to get OS info).
- To compile a Nim module into a ``.js`` file use the ``js`` command; the
- default is a ``.js`` file that is supposed to be referenced in an ``.html``
- file. However, you can also run the code with `nodejs`:idx:
- (`<http://nodejs.org>`_)::
- nim js -d:nodejs -r examples/hallo.nim
- Interfacing
- ===========
- Nim offers bidirectional interfacing with the target backend. This means
- that you can call backend code from Nim and Nim code can be called by
- the backend code. Usually the direction of which calls which depends on your
- software architecture (is Nim your main program or is Nim providing a
- component?).
- Nim code calling the backend
- ----------------------------
- Nim code can interface with the backend through the `Foreign function
- interface <manual.html#foreign-function-interface>`_ mainly through the
- `importc pragma <manual.html#foreign-function-interface-importc-pragma>`_.
- The ``importc`` pragma is the *generic* way of making backend symbols available
- in Nim and is available in all the target backends (JavaScript too). The C++
- or Objective-C backends have their respective `ImportCpp
- <manual.html#implementation-specific-pragmas-importcpp-pragma>`_ and
- `ImportObjC <manual.html#implementation-specific-pragmas-importobjc-pragma>`_
- pragmas to call methods from classes.
- Whenever you use any of these pragmas you need to integrate native code into
- your final binary. In the case of JavaScript this is no problem at all, the
- same HTML file which hosts the generated JavaScript will likely provide other
- JavaScript functions which you are importing with ``importc``.
- However, for the C like targets you need to link external code either
- statically or dynamically. The preferred way of integrating native code is to
- use dynamic linking because it allows you to compile Nim programs without
- the need for having the related development libraries installed. This is done
- through the `dynlib pragma for import
- <manual.html#foreign-function-interface-dynlib-pragma-for-import>`_, though
- more specific control can be gained using the `dynlib module <dynlib.html>`_.
- The `dynlibOverride <nimc.html#dynliboverride>`_ command line switch allows
- to avoid dynamic linking if you need to statically link something instead.
- Nim wrappers designed to statically link source files can use the `compile
- pragma <manual.html#implementation-specific-pragmas-compile-pragma>`_ if
- there are few sources or providing them along the Nim code is easier than using
- a system library. Libraries installed on the host system can be linked in with
- the `PassL pragma <manual.html#implementation-specific-pragmas-passl-pragma>`_.
- To wrap native code, take a look at the `c2nim tool <https://github.com/nim-lang/c2nim/blob/master/doc/c2nim.rst>`_ which helps
- with the process of scanning and transforming header files into a Nim
- interface.
- C invocation example
- ~~~~~~~~~~~~~~~~~~~~
- Create a ``logic.c`` file with the following content:
- .. code-block:: c
- int addTwoIntegers(int a, int b)
- {
- return a + b;
- }
- Create a ``calculator.nim`` file with the following content:
- .. code-block:: nim
- {.compile: "logic.c".}
- proc addTwoIntegers(a, b: cint): cint {.importc.}
- when isMainModule:
- echo addTwoIntegers(3, 7)
- With these two files in place, you can run ``nim c -r calculator.nim`` and
- the Nim compiler will compile the ``logic.c`` file in addition to
- ``calculator.nim`` and link both into an executable, which outputs ``10`` when
- run. Another way to link the C file statically and get the same effect would
- be to remove the line with the ``compile`` pragma and run the following typical
- Unix commands::
- $ gcc -c logic.c
- $ ar rvs mylib.a logic.o
- $ nim c --passL:mylib.a -r calculator.nim
- Just like in this example we pass the path to the ``mylib.a`` library (and we
- could as well pass ``logic.o``) we could be passing switches to link any other
- static C library.
- JavaScript invocation example
- ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- Create a ``host.html`` file with the following content:
- .. code-block::
- <html><body>
- <script type="text/javascript">
- function addTwoIntegers(a, b)
- {
- return a + b;
- }
- </script>
- <script type="text/javascript" src="calculator.js"></script>
- </body></html>
- Create a ``calculator.nim`` file with the following content (or reuse the one
- from the previous section):
- .. code-block:: nim
- proc addTwoIntegers(a, b: int): int {.importc.}
- when isMainModule:
- echo addTwoIntegers(3, 7)
- Compile the Nim code to JavaScript with ``nim js -o:calculator.js
- calculator.nim`` and open ``host.html`` in a browser. If the browser supports
- javascript, you should see the value ``10`` in the browser's console. Use the
- `dom module <dom.html>`_ for specific DOM querying and modification procs
- or take a look at `karax <https://github.com/pragmagic/karax>`_ for how to
- develop browser-based applications.
- Backend code calling Nim
- ------------------------
- Backend code can interface with Nim code exposed through the `exportc
- pragma <manual.html#foreign-function-interface-exportc-pragma>`_. The
- ``exportc`` pragma is the *generic* way of making Nim symbols available to
- the backends. By default, the Nim compiler will mangle all the Nim symbols to
- avoid any name collision, so the most significant thing the ``exportc`` pragma
- does is maintain the Nim symbol name, or if specified, use an alternative
- symbol for the backend in case the symbol rules don't match.
- The JavaScript target doesn't have any further interfacing considerations
- since it also has garbage collection, but the C targets require you to
- initialize Nim's internals, which is done calling a ``NimMain`` function.
- Also, C code requires you to specify a forward declaration for functions or
- the compiler will assume certain types for the return value and parameters
- which will likely make your program crash at runtime.
- The Nim compiler can generate a C interface header through the ``--header``
- command-line switch. The generated header will contain all the exported
- symbols and the ``NimMain`` proc which you need to call before any other
- Nim code.
- Nim invocation example from C
- ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- Create a ``fib.nim`` file with the following content:
- .. code-block:: nim
- proc fib(a: cint): cint {.exportc.} =
- if a <= 2:
- result = 1
- else:
- result = fib(a - 1) + fib(a - 2)
- Create a ``maths.c`` file with the following content:
- .. code-block:: c
- #include "fib.h"
- #include <stdio.h>
- int main(void)
- {
- NimMain();
- for (int f = 0; f < 10; f++)
- printf("Fib of %d is %d\n", f, fib(f));
- return 0;
- }
- Now you can run the following Unix like commands to first generate C sources
- from the Nim code, then link them into a static binary along your main C
- program::
- $ nim c --noMain --noLinking --header:fib.h fib.nim
- $ gcc -o m -I$HOME/.cache/nim/fib_d -Ipath/to/nim/lib $HOME/.cache/nim/fib_d/*.c maths.c
- The first command runs the Nim compiler with three special options to avoid
- generating a ``main()`` function in the generated files, avoid linking the
- object files into a final binary, and explicitly generate a header file for C
- integration. All the generated files are placed into the ``nimcache``
- directory. That's why the next command compiles the ``maths.c`` source plus
- all the ``.c`` files from ``nimcache``. In addition to this path, you also
- have to tell the C compiler where to find Nim's ``nimbase.h`` header file.
- Instead of depending on the generation of the individual ``.c`` files you can
- also ask the Nim compiler to generate a statically linked library::
- $ nim c --app:staticLib --noMain --header fib.nim
- $ gcc -o m -Inimcache -Ipath/to/nim/lib libfib.nim.a maths.c
- The Nim compiler will handle linking the source files generated in the
- ``nimcache`` directory into the ``libfib.nim.a`` static library, which you can
- then link into your C program. Note that these commands are generic and will
- vary for each system. For instance, on Linux systems you will likely need to
- use ``-ldl`` too to link in required dlopen functionality.
- Nim invocation example from JavaScript
- ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
- Create a ``mhost.html`` file with the following content:
- .. code-block::
- <html><body>
- <script type="text/javascript" src="fib.js"></script>
- <script type="text/javascript">
- alert("Fib for 9 is " + fib(9));
- </script>
- </body></html>
- Create a ``fib.nim`` file with the following content (or reuse the one
- from the previous section):
- .. code-block:: nim
- proc fib(a: cint): cint {.exportc.} =
- if a <= 2:
- result = 1
- else:
- result = fib(a - 1) + fib(a - 2)
- Compile the Nim code to JavaScript with ``nim js -o:fib.js fib.nim`` and
- open ``mhost.html`` in a browser. If the browser supports javascript, you
- should see an alert box displaying the text ``Fib for 9 is 34``. As mentioned
- earlier, JavaScript doesn't require an initialization call to ``NimMain`` or
- a similar function and you can call the exported Nim proc directly.
- Nimcache naming logic
- ---------------------
- The `nimcache`:idx: directory is generated during compilation and will hold
- either temporary or final files depending on your backend target. The default
- name for the directory depends on the used backend and on your OS but you can
- use the ``--nimcache`` `compiler switch
- <nimc.html#compiler-usage-commandminusline-switches>`_ to change it.
- Memory management
- =================
- In the previous sections, the ``NimMain()`` function reared its head. Since
- JavaScript already provides automatic memory management, you can freely pass
- objects between the two languages without problems. In C and derivate languages
- you need to be careful about what you do and how you share memory. The
- previous examples only dealt with simple scalar values, but passing a Nim
- string to C, or reading back a C string in Nim already requires you to be
- aware of who controls what to avoid crashing.
- Strings and C strings
- ---------------------
- The manual mentions that `Nim strings are implicitly convertible to
- cstrings <manual.html#types-cstring-type>`_ which makes interaction usually
- painless. Most C functions accepting a Nim string converted to a
- ``cstring`` will likely not need to keep this string around and by the time
- they return the string won't be needed anymore. However, for the rare cases
- where a Nim string has to be preserved and made available to the C backend
- as a ``cstring``, you will need to manually prevent the string data from being
- freed with `GC_ref <system.html#GC_ref,string>`_ and `GC_unref
- <system.html#GC_unref,string>`_.
- A similar thing happens with C code invoking Nim code which returns a
- ``cstring``. Consider the following proc:
- .. code-block:: nim
- proc gimme(): cstring {.exportc.} =
- result = "Hey there C code! " & $rand(100)
- Since Nim's garbage collector is not aware of the C code, once the
- ``gimme`` proc has finished it can reclaim the memory of the ``cstring``.
- However, from a practical standpoint, the C code invoking the ``gimme``
- function directly will be able to use it since Nim's garbage collector has
- not had a chance to run *yet*. This gives you enough time to make a copy for
- the C side of the program, as calling any further Nim procs *might* trigger
- garbage collection making the previously returned string garbage. Or maybe you
- are `yourself triggering the collection <gc.html>`_.
- Custom data types
- -----------------
- Just like strings, custom data types that are to be shared between Nim and
- the backend will need careful consideration of who controls who. If you want
- to hand a Nim reference to C code, you will need to use `GC_ref
- <system.html#GC_ref,ref.T>`_ to mark the reference as used, so it does not get
- freed. And for the C backend you will need to expose the `GC_unref
- <system.html#GC_unref,ref.T>`_ proc to clean up this memory when it is not
- required anymore.
- Again, if you are wrapping a library which *mallocs* and *frees* data
- structures, you need to expose the appropriate *free* function to Nim so
- you can clean it up. And of course, once cleaned you should avoid accessing it
- from Nim (or C for that matter). Typically C data structures have their own
- ``malloc_structure`` and ``free_structure`` specific functions, so wrapping
- these for the Nim side should be enough.
- Thread coordination
- -------------------
- When the ``NimMain()`` function is called Nim initializes the garbage
- collector to the current thread, which is usually the main thread of your
- application. If your C code later spawns a different thread and calls Nim
- code, the garbage collector will fail to work properly and you will crash.
- As long as you don't use the threadvar emulation Nim uses native thread
- variables, of which you get a fresh version whenever you create a thread. You
- can then attach a GC to this thread via
- .. code-block:: nim
- system.setupForeignThreadGc()
- It is **not** safe to disable the garbage collector and enable it after the
- call from your background thread even if the code you are calling is short
- lived.
- Before the thread exits, you should tear down the thread's GC to prevent memory
- leaks by calling
- .. code-block:: nim
- system.tearDownForeignThreadGc()
|