56c3d9f9c0
According to v8.h, "the returned handle is valid until this TryCatch block has been destroyed". This is incorrect, as can be demonstrated with the test below. In practice the return value lives no longer and no shorter than the active HandleScope at the time these methods are called. An issue has been opened about this in the V8 bug tracker: https://bugs.chromium.org/p/v8/issues/detail?id=10537. ```rust fn try_catch_bad_lifetimes() { let _setup_guard = setup(); let mut isolate = v8::Isolate::new(Default::default()); let mut hs = v8::HandleScope::new(&mut isolate); let scope = hs.enter(); let context = v8::Context::new(scope); let mut cs = v8::ContextScope::new(scope, context); let scope = cs.enter(); let caught_msg_2 = { let mut try_catch = v8::TryCatch::new(scope); let try_catch = try_catch.enter(); let caught_msg_1 = { let mut hs = v8::HandleScope::new(scope); let scope = hs.enter(); // Throw exception #1. let msg_1 = v8::String::new(scope, "BOOM!").unwrap(); let exc_1 = v8::Exception::type_error(scope, msg_1); scope.isolate().throw_exception(exc_1); // Catch exception #1. let caught_msg_1 = try_catch.message().unwrap(); let caught_str_1 = caught_msg_1.get(scope).to_rust_string_lossy(scope); assert!(caught_str_1.contains("BOOM")); // Move `caught_msg_1` out of the HandleScope it was created in. // The borrow checker allows this because `caught_msg_1`'s // lifetime is contrained to not outlive the TryCatch, but it is // allowed to outlive the HandleScope that was active when the // exception was caught. caught_msg_1 }; // Next line crashes. let caught_str_1 = caught_msg_1.get(scope).to_rust_string_lossy(scope); assert!(caught_str_1.contains("BOOM")); // Throws exception #2. let msg_2 = v8::String::new(scope, "DANG!").unwrap(); let exc_2 = v8::Exception::type_error(scope, msg_2); scope.isolate().throw_exception(exc_2); // Catch exception #2. let caught_msg_2 = try_catch.message().unwrap(); let caught_str_2 = caught_msg_2.get(scope).to_rust_string_lossy(scope); assert!(caught_str_2.contains("DANG")); // Move `caught_msg_2` out of the extent of the TryCatch, but still // within the extent of its HandleScope. This is unnecessarily // rejected at compile time. caught_msg_2 }; let caught_str_2 = caught_msg_2.get(scope).to_rust_string_lossy(scope); assert!(caught_str_2.contains("DANG")); } ``` |
||
---|---|---|
.github/workflows | ||
base/trace_event | ||
build@17f96622d0 | ||
buildtools@4712bad481 | ||
src | ||
tests | ||
third_party | ||
tools | ||
v8@119b195598 | ||
.clang-format | ||
.gitignore | ||
.gitmodules | ||
.gn | ||
.prettierrc.json | ||
.rustfmt.toml | ||
BUILD.gn | ||
build.rs | ||
Cargo.lock | ||
Cargo.toml | ||
LICENSE | ||
README.md |
Rusty V8 Binding
V8 Version: 8.4.300, 2020-05-05
Goals
-
Provide high quality Rust bindings to V8's C++ API. The API should match the original API as closely as possible.
-
Do not introduce additional call overhead. (For example, previous attempts at Rust V8 bindings forced the use of Persistent handles.)
-
Do not rely on a binary
libv8.a
built outside of cargo. V8 is a very large project (over 600,000 lines of C++) which often takes 30 minutes to compile. Furthermore, V8 relies on Chromium's bespoke build system (gn + ninja) which is not easy to use outside of Chromium. For this reason many attempts to bind to V8 rely on pre-built binaries that are built separately from the binding itself. While this is simple, it makes upgrading V8 difficult, it makes CI difficult, it makes producing builds with different configurations difficult, and it is a security concern since binary blobs can hide malicious code. For this reason we believe it is imperative to build V8 from source code during "cargo build". -
Publish the crate on crates.io and allow docs.rs to generate documentation. Due to the complexity and size of V8's build, this is nontrivial. For example the crate size must be kept under 10 MiB in order to publish.
Binary Build
V8 is very large and takes a long time to compile. Many users will prefer to use a prebuilt version of V8. We publish static libs for every version of rusty v8 on Github.
Binaries builds are turned on by default: cargo build
will initiate a download
from github to get the static lib. To disable this build using the
V8_FROM_SOURCE
environmental variable.
When making changes to rusty_v8 itself, it should be tested by build from source. The CI always builds from source.
Build V8 from Source
Use V8_FROM_SOURCE=1 cargo build -vv
to build the crate completely from
source.
The build scripts on Python 2.7, not Python 3. Do not open issues with us regarding Python 3; it is a non-trivial problem that must be fixed in Chromium..
For linux builds: glib-2.0 development files need to be installed such that
pkg-config can find them. On Ubuntu, run sudo apt install libglib2.0-dev
to
install them.
For Windows builds: the 64-bit toolchain needs to be used. 32-bit targets are not supported.
The build depends on several binary tools: gn
, ninja
and a recent version
of clang
(V8 relies on bleeding edge features). Because these are not
generally available they are automatically downloaded during the build by default.
It should be possible to opt out of the gn and ninja download by specifying the
$GN
and $NINJA
environmental variables. The clang download can be skipped by
providing a $CLANG_BASE_PATH
environmental variable pointing to a recent
llvm
/clang
installation (currently LLVM v8.0+ or Apple clang v11.0+).
You could also pass in additional arguments to gn
by setting the $GN_ARGS
environmental variable.
Env vars used in when building from source: SCCACHE
, GN
, NINJA
,
CLANG_BASE_PATH
, GN_ARGS
FAQ
Building V8 takes over 30 minutes, this is too slow for me to use this crate. What should I do?
Install sccache. Our build scripts will
detect and use sccache. Set the $SCCACHE
environmental variable if it's not in
your path.
What are all these random directories for like build
and buildtools
are
these really necessary?
In order to build V8 from source code, we must provide a certain directory structure with some git submodules from Chromium. We welcome any simplifications to the code base, but this is a structure we have found after many failed attempts that carefully balances the requirements of cargo crates and GN/Ninja.
V8 has a very large API with hundreds of methods. Why don't you automate the generation of this binding code?
In the limit we would like to auto-generate bindings. We have actually started down this route several times, however due to many eccentric features of the V8 API, this has not proven successful. Therefore we are proceeding in a brute-force fashion for now, focusing on solving our stated goals first. We hope to auto-generate bindings in the future.
Why are you building this?
This is to support the Deno project. We previously have gotten away with a simpler high-level Rust binding to V8 called libdeno. But as Deno has matured we've found ourselves continually needing access to an increasing amount of V8's API in Rust.
When building I get unknown argument: '-gno-inline-line-tables'
Use export GN_ARGS="no_inline_line_tables=false"
during build.