2024-06-10 09:11:44 -07:00
|
|
|
# Tomo Library/Module Design
|
|
|
|
|
|
|
|
There are two ways to "import" code that is defined elsewhere: local files from
|
|
|
|
the same project and shared library objects from another project. The first
|
|
|
|
type of import (local files) is necessary for splitting large projects into
|
|
|
|
smaller components for ease of understanding and compilation speed. The second
|
|
|
|
type of import (shared libraries) is to allow you to install third party
|
|
|
|
libraries or frameworks that can be used across many projects.
|
|
|
|
|
|
|
|
## Local Imports
|
|
|
|
|
|
|
|
To see how local imports work, let's look at a simple file:
|
|
|
|
|
|
|
|
```
|
|
|
|
// File: foo.tm
|
2024-08-15 22:20:46 -07:00
|
|
|
my_variable := "hello"
|
2024-06-10 09:11:44 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
When this file is compiled to a static object file by `tomo -c foo.tm`, it
|
|
|
|
produces the following C header file and C source file:
|
|
|
|
|
|
|
|
```c
|
|
|
|
// File: foo.tm.h
|
|
|
|
#pragma once
|
|
|
|
#include <tomo/tomo.h>
|
|
|
|
|
2024-08-15 22:20:46 -07:00
|
|
|
extern Text_t foo$my_variable;
|
2024-06-10 09:11:44 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
```c
|
|
|
|
// File: foo.tm.c
|
|
|
|
#include <tomo/tomo.h>
|
|
|
|
#include "foo.tm.h"
|
|
|
|
|
2024-08-15 22:20:46 -07:00
|
|
|
Text_t foo$my_variable = "hello";
|
2024-06-10 09:11:44 -07:00
|
|
|
```
|
|
|
|
|
|
|
|
Notice that the symbols defined here (`foo$my_variable`) use a file-based
|
|
|
|
prefix that includes a dollar sign. C compilers support an extension that
|
|
|
|
allows dollar signs in identifiers, and this allows us to use guaranteed-unique
|
|
|
|
prefixes so symbols from one file don't have naming collisions with symbols
|
|
|
|
in another file.
|
|
|
|
|
|
|
|
The C file is compiled by invoking the C compiler with something like: `cc
|
|
|
|
<flags...> -c foo.tm.c -o foo.tm.o`
|
|
|
|
|
|
|
|
Now, what happens if we want to _use_ the compiled object file?
|
|
|
|
|
|
|
|
```
|
|
|
|
// File: baz.tm
|
2024-08-22 11:02:48 -07:00
|
|
|
foo := use ./foo.tm
|
2024-06-10 09:11:44 -07:00
|
|
|
|
|
|
|
func say_stuff():
|
2025-01-12 13:49:58 -08:00
|
|
|
say("I got $(foo.my_variable) from foo")
|
2024-06-10 09:11:44 -07:00
|
|
|
|
|
|
|
func main():
|
|
|
|
say_stuff()
|
|
|
|
```
|
|
|
|
|
|
|
|
If I want to run `baz.tm` with `tomo baz.tm` then this transpiles to:
|
|
|
|
|
|
|
|
```c
|
|
|
|
// File: baz.tm.h
|
|
|
|
#pragma once
|
|
|
|
#include <tomo/tomo.h>
|
|
|
|
#include "./foo.tm.h"
|
|
|
|
|
|
|
|
void baz$say_stuff();
|
|
|
|
void baz$main();
|
|
|
|
```
|
|
|
|
|
|
|
|
```c
|
|
|
|
// File: baz.tm.c
|
|
|
|
#include <tomo/tomo.h>
|
|
|
|
#include "baz.tm.h"
|
|
|
|
|
|
|
|
public void baz$say_stuff()
|
|
|
|
{
|
2024-09-13 10:52:57 -07:00
|
|
|
say(Texts(Text("I got "), foo$my_variable, Text(" from foo")));
|
2024-06-10 09:11:44 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
public void baz$main()
|
|
|
|
{
|
|
|
|
baz$say_stuff();
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
Then `baz.tm.o` is compiled to a static object with `cc <flags...> -c baz.tm.c
|
|
|
|
-o baz.tm.o`.
|
|
|
|
|
|
|
|
Next, we need to create an actual executable file that will invoke `baz$main()`
|
|
|
|
(with any command line arguments). To do that, we create a small wrapper
|
|
|
|
program:
|
|
|
|
|
|
|
|
```c
|
|
|
|
// File: /tmp/program.c
|
|
|
|
#include <tomo/tomo.h>
|
|
|
|
#include "baz.tm.h"
|
|
|
|
|
|
|
|
int main(int argc, char *argv[])
|
|
|
|
{
|
|
|
|
tomo_init();
|
|
|
|
if (argc > 1)
|
|
|
|
errx(1, "This program doesn't take any arguments.");
|
|
|
|
baz$main();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
This program is compiled with the already-built object files to produce an
|
|
|
|
executable binary called `foo` like this: `cc <flags...> /tmp/program.c
|
|
|
|
foo.tm.o baz.tm.o -o baz`
|
|
|
|
|
|
|
|
Finally, the resulting binary can be executed to actually run the program!
|
|
|
|
|
|
|
|
|
|
|
|
## Shared Library Imports
|
|
|
|
|
2024-09-15 18:18:42 -07:00
|
|
|
In Tomo, a shared library is built out of a *directory* that contains multiple
|
|
|
|
`.tm` files. Each `.tm` file in the directory (excluding those that start with
|
|
|
|
an underscore) will be compiled and linked together to produce a single
|
2025-03-30 12:41:37 -07:00
|
|
|
`libwhatever.so` file (or `libwhatever.dylib` on Mac) and `whatever.h` file
|
|
|
|
that can be used by other Tomo projects. You can build a library by running
|
|
|
|
`tomo -L dirname/` or `tomo -L` in the current directory.
|
2024-06-10 09:11:44 -07:00
|
|
|
|
|
|
|
### Installing
|
|
|
|
|
2024-09-22 11:59:40 -07:00
|
|
|
If you additionally add the `-I` flag, Tomo will copy the entire directory
|
|
|
|
(excluding files and directories that begin with `.` such as `.git`) into
|
|
|
|
`~/.local/share/tomo/installed/` and create a symbolic link for the library's
|
2025-03-30 12:41:37 -07:00
|
|
|
`.so` file (or `.dylib` file on Mac) in `~/.local/share/tomo/lib/`.
|
2024-06-10 09:11:44 -07:00
|
|
|
|
|
|
|
### Using Shared Libraries
|
|
|
|
|
2024-09-15 18:25:43 -07:00
|
|
|
To use a shared library, write a statement like `use foo` with an unqualified
|
|
|
|
name (i.e. not an absolute or relative path like `/foo` or `./foo`). When a
|
2024-06-10 09:11:44 -07:00
|
|
|
program uses a shared library, that shared library gets dynamically linked to
|
|
|
|
the executable when compiling, and all of the necessary symbol information is
|
|
|
|
read from the source files during compilation.
|