Skip to content

Commit 4ab3361

Browse files
committed
Add Using Generated Bindings section
1 parent a55e696 commit 4ab3361

File tree

12 files changed

+154
-28
lines changed

12 files changed

+154
-28
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,4 @@
11
target/
22
scripts/.coursier
33
scripts/.scalafmt-*
4+
/docs/_book/

build.sbt

+4-3
Original file line numberDiff line numberDiff line change
@@ -91,10 +91,11 @@ lazy val tools = project in file("tools")
9191

9292
lazy val docs = project
9393
.in(file("docs"))
94-
.enablePlugins(ParadoxPlugin)
94+
.enablePlugins(GhpagesPlugin, ParadoxSitePlugin)
9595
.settings(
9696
paradoxTheme := Some(builtinParadoxTheme("generic")),
97-
paradoxProperties in Compile ++= Map(
97+
paradoxProperties in Paradox ++= Map(
9898
"github.base_url" -> "https://github.com/kornilova-l/scala-native-bindgen/tree/master/"
99-
)
99+
),
100+
git.remoteRepo := "git@github.com:kornilova-l/scala-native-bindgen.git"
100101
)

docs/src/main/paradox/obtaining-bindgen/index.md

-9
This file was deleted.

docs/src/main/paradox/command-line-usage/index.md docs/src/paradox/command-line-usage/index.md

+9-9
Original file line numberDiff line numberDiff line change
@@ -14,12 +14,12 @@ scala-native-bindgen --name uv /usr/include/uv.h -- > uv.scala
1414

1515
## Bindgen Options
1616

17-
| Option | Description |
18-
|----------------------|--------------------------------------------------------------------------------------------------------------|
19-
| `--link` | Library to link with, e.g. `--link` uv for libuv. |
20-
| `--no-link` | Library is static and does not require linking. |
21-
| `--name` | Scala object name that contains bindings. If `--no-link` is specified then `name` should match library name. |
22-
| `--package` | Package name of generated Scala file. |
23-
| `--exclude-prefix` | Functions and unused typedefs will be removed if their names have given prefix. |
24-
| `--extra-arg` | Additional argument to append to the compiler command line. |
25-
| `--extra-arg-before` | Additional argument to prepend to the compiler command line. |
17+
| Option | Description |
18+
|----------------------|---------------------------------------------------------------------------------|
19+
| `--link` | Library to link with, e.g. `--link` uv for libuv. |
20+
| `--no-link` | Library does not require linking. |
21+
| `--name` | Scala object name that contains bindings. Default value set to library name. |
22+
| `--package` | Package name of generated Scala file. |
23+
| `--exclude-prefix` | Functions and unused typedefs will be removed if their names have given prefix. |
24+
| `--extra-arg` | Additional argument to append to the compiler command line. |
25+
| `--extra-arg-before` | Additional argument to prepend to the compiler command line. |

docs/src/main/paradox/index.md docs/src/paradox/index.md

+3-2
Original file line numberDiff line numberDiff line change
@@ -2,9 +2,10 @@
22

33
@@@ index
44

5-
* [Obtaining bindgen](obtaining-bindgen/index.md)
6-
* [Usage](command-line-usage/index.md)
5+
* [Obtaining Bindgen](obtaining-bindgen/index.md)
6+
* [Command Line Usage](command-line-usage/index.md)
77
* [Limitations](limitations/index.md)
8+
* [Using Generated Bindings](using-generated-bindings/README.md)
89

910
@@@
1011

docs/src/main/paradox/obtaining-bindgen/docker-container.md docs/src/paradox/obtaining-bindgen/docker-container.md

+6-4
Original file line numberDiff line numberDiff line change
@@ -18,13 +18,15 @@ docker run -v "$(pwd)":/src -v /usr/include:/usr/include \
1818

1919
The docker image does not contain standard headers so it is important to
2020
mount all system include directories that are used by the header file
21-
passed to `scala-native-bindgen`. See the @github[docker-bindgen.sh](/scripts/docker-bindgen.sh) script for
22-
how to wrap the dockerized program. The `$CWD` of the container is
23-
`/src` which should be mounted from `$(pwd)` in case relative paths are
24-
used.
21+
passed to `scala-native-bindgen`.
22+
23+
See the [docker-bindgen.sh] script for how to wrap the dockerized program.
24+
The `$CWD` of the container is `/src` which should be mounted from `$(pwd)`
25+
in case relative paths are used.
2526

2627
Note, the `scalabindgen/scala-native-bindgen:latest` image is updated on
2728
each merge to the `master` branch.
2829

2930
[Docker]: https://www.docker.com/
31+
[docker-bindgen.sh]: https://github.com/kornilova-l/scala-native-bindgen/blob/master/scripts/docker-bindgen.sh
3032

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,17 @@
1+
# Obtaining Bindgen
2+
3+
@@@ index
4+
5+
* [Use Docker Container](docker-container.md)
6+
* [Build Binary with CMake](cmake.md)
7+
* [Build Binary with docker-compose](docker-compose.md)
8+
9+
@@@
10+
11+
There are 3 ways to obtain bindgen:
12+
13+
* @ref:[Use docker container](docker-container.md)
14+
15+
* @ref:[Build binary with CMake](cmake.md)
16+
17+
* @ref:[Build binary with docker-compose](docker-compose.md)
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,112 @@
1+
# Using Generated Bindings
2+
3+
Consider following header file:
4+
5+
```c
6+
struct point {
7+
float x;
8+
float y;
9+
};
10+
11+
struct vector {
12+
struct point a;
13+
struct point b;
14+
};
15+
16+
struct vector *add(struct vector *v1, struct vector *v2);
17+
```
18+
19+
Bindgen will generate type aliases for the structs, binding for function `add`
20+
and helper functions that make usage of structs easier.
21+
```scala
22+
import scala.scalanative._
23+
import scala.scalanative.native._
24+
25+
@native.link("mylib")
26+
@native.extern
27+
object mylib {
28+
type struct_point = native.CStruct2[native.CFloat, native.CFloat]
29+
type struct_vector = native.CStruct2[struct_point, struct_point]
30+
def add(v1: native.Ptr[struct_vector], v2: native.Ptr[struct_vector]): native.Ptr[struct_vector] = native.extern
31+
}
32+
33+
import mylib._
34+
35+
object mylibHelpers {
36+
37+
implicit class struct_point_ops(val p: native.Ptr[struct_point]) extends AnyVal {
38+
def x: native.CFloat = !p._1
39+
def x_=(value: native.CFloat): Unit = !p._1 = value
40+
def y: native.CFloat = !p._2
41+
def y_=(value: native.CFloat): Unit = !p._2 = value
42+
}
43+
44+
def struct_point()(implicit z: native.Zone): native.Ptr[struct_point] = native.alloc[struct_point]
45+
46+
implicit class struct_vector_ops(val p: native.Ptr[struct_vector]) extends AnyVal {
47+
def a: native.Ptr[struct_point] = p._1
48+
def a_=(value: native.Ptr[struct_point]): Unit = !p._1 = !value
49+
def b: native.Ptr[struct_point] = p._2
50+
def b_=(value: native.Ptr[struct_point]): Unit = !p._2 = !value
51+
}
52+
53+
def struct_vector()(implicit z: native.Zone): native.Ptr[struct_vector] = native.alloc[struct_vector]
54+
}
55+
```
56+
Let's write code that creates two vectors, adds them and prints resulting
57+
vector.
58+
59+
First we need to create points for vectors. We will use `native.Zone` to
60+
allocate struct (more information on memory management can be found
61+
here: [Scala Native memory management]).
62+
63+
Helper object `mylibHelpers` contains function for struct allocation.
64+
To import it use `import mylibHelpers._`
65+
66+
Let's create points for first vector:
67+
```scala
68+
import mylibHelpers._
69+
import scala.scalanative.native.Zone
70+
71+
object Hello extends App {
72+
Zone { implicit zone =>
73+
val vec1p1 = struct_point()
74+
val vec1p2 = struct_point()
75+
}
76+
}
77+
```
78+
79+
Now we want to set fields of created points. Scala Native provides access
80+
to fields by using `_N` methods where `N` is index of a field
81+
(see [Scala Native memory layout types]).
82+
83+
Bindgen generates implicit helper classes that wrap calls to `_N` in functions
84+
with meaningful names. We already imported helper class, so we can use the
85+
functions:
86+
```scala
87+
vec1p1.x = 0
88+
vec1p1.y = 1
89+
90+
vec1p2.x = 6
91+
vec1p2.y = 3
92+
```
93+
94+
Lets create first vector. Note that `struct_vector` contains
95+
fields of type `struct_point` but setters accept variables of type
96+
`native.Ptr[struct_point]`. It helps to avoid Scala Native limitation that
97+
does not allow passing structs and arrays by value
98+
(see @github[scala-native/scala-native#555](scala-native/scala-native#555)).
99+
```scala
100+
val vec1 = struct_vector()
101+
vec1.a = vec1p1
102+
vec1.b = vec1p2
103+
```
104+
Repeat these steps to create second vector. Once both vectors are created we can
105+
call `add` function and print the result:
106+
```scala
107+
val vec3 = mylib.add(vec1, vec2)
108+
println(s"(${vec3.a.x}, ${vec3.a.y}), (${vec3.b.x}, ${vec3.b.y})")
109+
```
110+
111+
[Scala Native memory management]: http://www.scala-native.org/en/latest/user/interop.html#memory-management
112+
[Scala Native memory layout types]: http://www.scala-native.org/en/latest/user/interop.html#memory-layout-types

project/plugins.sbt

+2-1
Original file line numberDiff line numberDiff line change
@@ -1,2 +1,3 @@
11
addSbtPlugin("org.scala-native" % "sbt-scala-native" % "0.3.7")
2-
addSbtPlugin("com.lightbend.paradox" % "sbt-paradox" % "0.3.5")
2+
addSbtPlugin("com.typesafe.sbt" % "sbt-site" % "1.3.2")
3+
addSbtPlugin("com.typesafe.sbt" % "sbt-ghpages" % "0.6.2")

0 commit comments

Comments
 (0)