Skip to content

Commit 61a45e8

Browse files
committed
Add example of linking against a C++ shared library
1 parent dfbf6b9 commit 61a45e8

File tree

2 files changed

+158
-0
lines changed

2 files changed

+158
-0
lines changed

src/development_tools/build_tools.md

+2
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,8 @@ See crates.io's [documentation on the matter][build-script-docs] for more inform
1111

1212
{{#include build_tools/cc-defines.md}}
1313

14+
{{#include build_tools/cc-shared-library.md}}
15+
1416
{{#include ../links.md}}
1517

1618
[build-script-docs]: http://doc.crates.io/build-script.html
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,156 @@
1+
## Compile and link statically to a bundled C library
2+
3+
[![cc-badge]][cc] [![cat-development-tools-badge]][cat-development-tools]
4+
5+
To accommodate more complex scenarios where your rust project needs to integrate
6+
with a shared library that will be dynamically linked at runtime, we can use a
7+
mixture of build scripts (**build.rs**) and compiler directives to teach the
8+
compiler and the final binary how to find and link to the shared library. This
9+
is useful when embedding a system-level library that aren't intended to be
10+
compiled into their client applications directly, or a library that gets updated
11+
regularly outside the scope of your project.
12+
13+
This example does not use the [**cc**][cc] crate to compile the shared library,
14+
since you often don't have access to a shared library's source or build
15+
process. Rather, you're just consuming the library's binary directly.
16+
17+
For the purpose of this example, the shared library's source is in
18+
**src/mylibrary.h** and **src/mylibrary.cc** so we can transparently see its API
19+
surface. But you can build and distribute the resulting shared library binary
20+
independent of your rust project's build process. This example also provides a
21+
simple **src/Makefile** to show how you might build this simple library outside
22+
of our our rust build process, to produce a suitable binary that our project can
23+
consume.
24+
25+
Before compiling rust source code, the "build" file (**build.rs**) specified in
26+
**Cargo.toml** runs. It enumerates the shared libraries that our project depends
27+
on with the [`cargo:rustc-link-lib`][rustc-link-lib] directive, and tells
28+
**rustc** how to find those libraries at compile time with the
29+
[`cargo:rustc-link-search`][rustc-link-search] directive.
30+
31+
The [`cargo:rustc-link-lib`][rustc-link-lib] directive also bakes dynamic
32+
dependency metadata into the final binary, which tells the linker the names of
33+
the libraries to find at runtime. At runtime, it will search for these libraries
34+
in its default places (usually system defaults like `/usr/lib`), but you can
35+
also include specific runtime path information in the final binary with the
36+
[`cargo:rustc-link-arg][rustc-link-arg] directive, which informs the linker
37+
*where* to search for shared library dependencies. This can make your binary
38+
more portable, but we do not do this in the simple example below.
39+
40+
### `Cargo.toml`
41+
42+
```toml
43+
[package]
44+
...
45+
build = "build.rs"
46+
```
47+
48+
### `build.rs`
49+
50+
```rust,edition2021,no_run
51+
fn main() {
52+
// At compile time, we specify dynamic library dependency entries to bake
53+
// into the binary for the linker to consult at runtime.
54+
println!("cargo:rustc-link-search=native=.");
55+
println!("cargo:rustc-link-lib=dylib=mylibrary");
56+
}
57+
```
58+
59+
### `src/library.c`
60+
61+
```cpp
62+
#include "api.h"
63+
64+
#include <iostream>
65+
66+
extern "C" {
67+
68+
int add_numbers(int a, int b) {
69+
return a + b;
70+
}
71+
72+
void greet(const char* string) {
73+
std::cout << "Hello from a C shared library, " << string << std::endl;
74+
}
75+
76+
} // extern "C"
77+
```
78+
79+
### `src/library.h`
80+
81+
```cpp
82+
// Use C linkage for ABI stability.
83+
extern "C" {
84+
85+
int add_numbers(int a, int b);
86+
void greet(const char* string);
87+
88+
} // extern "C"
89+
```
90+
91+
### `src/Makefile`
92+
93+
```Makefile
94+
CC=g++
95+
CFLAGS=--std=c++11 -Wall -Werror -I.
96+
97+
all: lib
98+
99+
lib: src/mylibrary.c src/mylibrary.h
100+
# Position-independent code (`-fpic`) is required for shared libraries.
101+
$(CC) -c -fpic src/mylibrary.c -o mylibrary.o $(CFLAGS)
102+
$(CC) -shared -o libmylibrary.so mylibrary.o
103+
rm mylibrary.o
104+
```
105+
106+
### `src/main.rs`
107+
108+
```rust,edition2021,ignore
109+
use std::ffi::CString;
110+
use std::os::raw::c_char;
111+
112+
// Compiler directive to tell the Rust which library contains the external
113+
// symbols we enumerate in this specific `extern` block. Here we specify a C
114+
// shared library called `libmylibrary`.
115+
#[link(name = "mylibrary")]
116+
extern "C" {
117+
fn add_numbers(a: i32, b: i32) -> i32;
118+
fn greet(name: *const c_char);
119+
}
120+
121+
fn prompt(s: &str) -> String {
122+
use std::io::Write;
123+
print!("{}", s);
124+
std::io::stdout().flush().unwrap();
125+
let mut input = String::new();
126+
std::io::stdin().read_line(&mut input).unwrap();
127+
input.trim().to_string()
128+
}
129+
130+
// Safe wrapper around the unsafe C add_numbers function.
131+
pub fn library_add_numbers(a: i32, b: i32) -> i32 {
132+
unsafe {
133+
add_numbers(a, b)
134+
}
135+
}
136+
137+
// Safe wrapper around the unsafe C add_numbers function.
138+
pub fn library_greet(name: CString) {
139+
unsafe {
140+
greet(name.as_ptr())
141+
}
142+
}
143+
144+
fn main() {
145+
let name = prompt("What's your name? ");
146+
let c_name = CString::new(name).unwrap();
147+
library_greet(c_name);
148+
let result = library_add_numbers(10, 12);
149+
println!("Addition result: {}", result);
150+
}
151+
```
152+
153+
[cc]: https://docs.rs/cc/latest/cc/
154+
[rustc-link-arg]: https://doc.rust-lang.org/cargo/reference/build-scripts.html#rustc-link-arg
155+
[rustc-link-lib]: https://doc.rust-lang.org/cargo/reference/build-scripts.html#rustc-link-lib
156+
[rustc-link-search]: https://doc.rust-lang.org/cargo/reference/build-scripts.html#rustc-link-search

0 commit comments

Comments
 (0)