mirror of
https://github.com/fluencelabs/wasm-bindgen
synced 2025-04-21 12:12:13 +00:00
* Gate `web-sys` APIs on activated features Currently the compile times of `web-sys` are unfortunately prohibitive, increasing the barrier to using it. This commit updates the crate to instead have all APIs gated by a set of Cargo features which affect what bindings are generated at compile time (and which are then compiled by rustc). It's significantly faster to activate only a handful of features vs all thousand of them! A magical env var is added to print the list of all features that should be generated, and then necessary logic is added to ferry features from the build script to the webidl crate which then uses that as a filter to remove items after parsing. Currently parsing is pretty speedy so we'll unconditionally parse all WebIDL files, but this may change in the future! For now this will make the `web-sys` crate a bit less ergonomic to use as lots of features will need to be specified, but it should make it much more approachable in terms of first-user experience with compile times. * Fix AppVeyor testing web-sys * FIx a typo * Udpate feature listings from rebase conflicts * Add some crate docs and such
136 lines
4.4 KiB
Rust
136 lines
4.4 KiB
Rust
use std::collections::hash_map::DefaultHasher;
|
|
use std::env;
|
|
use std::fmt;
|
|
use std::hash::{Hash, Hasher};
|
|
use std::iter::FromIterator;
|
|
use std::sync::atomic::Ordering::SeqCst;
|
|
use std::sync::atomic::{AtomicBool, ATOMIC_BOOL_INIT};
|
|
use std::sync::atomic::{AtomicUsize, ATOMIC_USIZE_INIT};
|
|
|
|
use ast;
|
|
use proc_macro2::{self, Ident};
|
|
use syn;
|
|
|
|
fn is_rust_keyword(name: &str) -> bool {
|
|
match name {
|
|
"abstract" | "alignof" | "as" | "become" | "box" | "break" | "const" | "continue"
|
|
| "crate" | "do" | "else" | "enum" | "extern" | "false" | "final" | "fn" | "for" | "if"
|
|
| "impl" | "in" | "let" | "loop" | "macro" | "match" | "mod" | "move" | "mut"
|
|
| "offsetof" | "override" | "priv" | "proc" | "pub" | "pure" | "ref" | "return"
|
|
| "Self" | "self" | "sizeof" | "static" | "struct" | "super" | "trait" | "true"
|
|
| "type" | "typeof" | "unsafe" | "unsized" | "use" | "virtual" | "where" | "while"
|
|
| "yield" | "bool" | "_" => true,
|
|
_ => false,
|
|
}
|
|
}
|
|
|
|
// Create an `Ident`, possibly mangling it if it conflicts with a Rust keyword.
|
|
pub fn rust_ident(name: &str) -> Ident {
|
|
if name == "" {
|
|
panic!("tried to create empty Ident (from \"\")");
|
|
} else if is_rust_keyword(name) {
|
|
Ident::new(&format!("{}_", name), proc_macro2::Span::call_site())
|
|
} else if name.chars().next().unwrap().is_ascii_digit() {
|
|
Ident::new(&format!("N{}", name), proc_macro2::Span::call_site())
|
|
} else {
|
|
raw_ident(name)
|
|
}
|
|
}
|
|
|
|
// Create an `Ident` without checking to see if it conflicts with a Rust
|
|
// keyword.
|
|
pub fn raw_ident(name: &str) -> Ident {
|
|
Ident::new(name, proc_macro2::Span::call_site())
|
|
}
|
|
|
|
/// Create a path type from the given segments. For example an iterator yielding
|
|
/// the idents `[foo, bar, baz]` will result in the path type `foo::bar::baz`.
|
|
pub fn simple_path_ty<I>(segments: I) -> syn::Type
|
|
where
|
|
I: IntoIterator<Item = Ident>,
|
|
{
|
|
path_ty(false, segments)
|
|
}
|
|
|
|
/// Create a global path type from the given segments. For example an iterator
|
|
/// yielding the idents `[foo, bar, baz]` will result in the path type
|
|
/// `::foo::bar::baz`.
|
|
pub fn leading_colon_path_ty<I>(segments: I) -> syn::Type
|
|
where
|
|
I: IntoIterator<Item = Ident>,
|
|
{
|
|
path_ty(true, segments)
|
|
}
|
|
|
|
fn path_ty<I>(leading_colon: bool, segments: I) -> syn::Type
|
|
where
|
|
I: IntoIterator<Item = Ident>,
|
|
{
|
|
let segments: Vec<_> = segments
|
|
.into_iter()
|
|
.map(|i| syn::PathSegment {
|
|
ident: i,
|
|
arguments: syn::PathArguments::None,
|
|
})
|
|
.collect();
|
|
|
|
syn::TypePath {
|
|
qself: None,
|
|
path: syn::Path {
|
|
leading_colon: if leading_colon {
|
|
Some(Default::default())
|
|
} else {
|
|
None
|
|
},
|
|
segments: syn::punctuated::Punctuated::from_iter(segments),
|
|
},
|
|
}.into()
|
|
}
|
|
|
|
pub fn ident_ty(ident: Ident) -> syn::Type {
|
|
simple_path_ty(Some(ident))
|
|
}
|
|
|
|
pub fn wrap_import_function(function: ast::ImportFunction) -> ast::Import {
|
|
ast::Import {
|
|
module: None,
|
|
js_namespace: None,
|
|
kind: ast::ImportKind::Function(function),
|
|
}
|
|
}
|
|
|
|
/// Small utility used when generating symbol names.
|
|
///
|
|
/// Hashes the public field here along with a few cargo-set env vars to
|
|
/// distinguish between runs of the procedural macro.
|
|
#[derive(Debug)]
|
|
pub struct ShortHash<T>(pub T);
|
|
|
|
impl<T: Hash> fmt::Display for ShortHash<T> {
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
|
static HASHED: AtomicBool = ATOMIC_BOOL_INIT;
|
|
static HASH: AtomicUsize = ATOMIC_USIZE_INIT;
|
|
|
|
// Try to amortize the cost of loading env vars a lot as we're gonna be
|
|
// hashing for a lot of symbols.
|
|
if !HASHED.load(SeqCst) {
|
|
let mut h = DefaultHasher::new();
|
|
env::var("CARGO_PKG_NAME")
|
|
.expect("should have CARGO_PKG_NAME env var")
|
|
.hash(&mut h);
|
|
env::var("CARGO_PKG_VERSION")
|
|
.expect("should have CARGO_PKG_VERSION env var")
|
|
.hash(&mut h);
|
|
// This may chop off 32 bits on 32-bit platforms, but that's ok, we
|
|
// just want something to mix in below anyway.
|
|
HASH.store(h.finish() as usize, SeqCst);
|
|
HASHED.store(true, SeqCst);
|
|
}
|
|
|
|
let mut h = DefaultHasher::new();
|
|
HASH.load(SeqCst).hash(&mut h);
|
|
self.0.hash(&mut h);
|
|
write!(f, "{:016x}", h.finish())
|
|
}
|
|
}
|