oss-fuzz/projects/libra
David Wong f5098035eb
[libra] Fix building failure (#3566)
After many days banging my head on FFI issues in rust,
I hereby present a fix to the issue.

Note that I've got some help, and I'm not sure I understand everything here.
But this is my understanding of what was not working, and how we fixed it.

The **problem** is that on Ubuntu 16 with llvm/clang 10,
we were **statically linking libc++** in [rocksdb][1]:

```rust
let stdlib = if tool.is_like_gnu() {
  "libstdc++.a"
} else if tool.is_like_clang() {
  "libc++.a"
} else {
  // Don't link to c++ statically on windows.
  return;
};

// ...

// remove lib prefix and .a postfix.
println!(
  "cargo:rustc-link-lib=static={}",
  &stdlib[3..stdlib.len() - 2]
);
```

This means that during building, when we reach building of rocksdb,
we import a number of symbols from libc++ (like [__muloti4][2])
that end up in the associated `.rlib` (rust obj file).

These symbols interestingly do not exist in libstdc++ which is used by gcc.
This is important because on linux (unlike mac), the rust toolchain is compiled with gcc.
So these intrinsics are not present in the linux rust toolchain,
and have been redeclared in the [compiler-builtins][3] crate.

So here is the problem:

* rust toolchain's defines these intrinsics functions
* libc++ defines these intrinsics functions

And the recipe for disaster:

* libc++ is statically linked in rocksdb, which means all the symbols are imported
* symbols in rocksdb's produced `.rlib` are colliding with the symbols from the rust toolchain `.rlib`

To fix this. Maybe we could have compiled the stuff with libstdc++?
But instead we:

1. removed the static linking
2. we linked libc++ at the very last moment via:
  ```rust
  RUSTFLAGS="-C link-arg=-L/usr/local/lib -C link-arg=-lc++"
  ```

At final linking time, the linker sees that the intrinsics are already defined in one of the `.rlib`
(produced by compiler-builtins) and so does not import these functions from libc++.
Actually, at this point it only statically link the functions that need to be linked.

It seems to work.

[1]: c79d2c2ac6/librocksdb_sys/build.rs (L115)
[2]: https://github.com/llvm-mirror/libcxx/blob/master/src/filesystem/int128_builtins.cpp
[3]: e578d47247/src/int/mul.rs (L107)
2020-04-02 14:28:09 -07:00
..
Dockerfile
build.sh [libra] Fix building failure (#3566) 2020-04-02 14:28:09 -07:00
project.yaml trying a distribution list as CC for libra (#3536) 2020-03-24 12:25:13 -07:00