// Copyright 2018-2019 the Deno authors. All rights reserved. MIT license. use cargo_gn; use std::env; use std::path::Path; use std::path::PathBuf; use std::process::exit; use std::process::Command; use which::which; fn main() { // Detect if trybuild tests are being compiled. let is_trybuild = env::var_os("DENO_TRYBUILD").is_some(); // Don't build if "cargo doc" is being run. This is to support docs.rs. let is_cargo_doc = env::var_os("RUSTDOCFLAGS").is_some(); if !(is_trybuild || is_cargo_doc) { build_v8() } if !is_cargo_doc { print_link_flags() } } fn build_v8() { env::set_var("DEPOT_TOOLS_WIN_TOOLCHAIN", "0"); // cargo publish doesn't like pyc files. env::set_var("PYTHONDONTWRITEBYTECODE", "1"); // git submodule update --init --recursive let libcxx_src = PathBuf::from("buildtools/third_party/libc++/trunk/src"); if !libcxx_src.is_dir() { eprintln!( "missing source code. Run 'git submodule update --init --recursive'" ); exit(1); } if need_gn_ninja_download() { download_gn_ninja_binaries(); } // On windows, rustc cannot link with a V8 debug build. let mut gn_args = if cargo_gn::is_debug() && !cfg!(target_os = "windows") { vec!["is_debug=true".to_string()] } else { vec!["is_debug=false".to_string()] }; let clang_base_path = clang_download(); gn_args.push(format!("clang_base_path={:?}", clang_base_path)); if let Some(p) = env::var_os("SCCACHE") { cc_wrapper(&mut gn_args, &Path::new(&p)); } else if let Ok(p) = which("sccache") { cc_wrapper(&mut gn_args, &p); } else { println!("cargo:warning=Not using sccache"); } let gn_root = env::var("CARGO_MANIFEST_DIR").unwrap(); let gn_out = cargo_gn::maybe_gen(&gn_root, gn_args); assert!(gn_out.exists()); assert!(gn_out.join("args.gn").exists()); cargo_gn::build("rusty_v8"); } fn platform() -> &'static str { #[cfg(target_os = "windows")] { "win" } #[cfg(target_os = "linux")] { "linux64" } #[cfg(target_os = "macos")] { "mac" } } fn download_gn_ninja_binaries() { let root = env::current_dir().unwrap(); let out_dir = root.join(env::var_os("OUT_DIR").unwrap()); let d = out_dir.join("gn_ninja_binaries").join(platform()); let gn = d.join("gn"); let ninja = d.join("ninja"); #[cfg(windows)] let gn = gn.with_extension("exe"); #[cfg(windows)] let ninja = ninja.with_extension("exe"); if !gn.exists() || !ninja.exists() { let status = Command::new("python") .arg("./tools/gn_ninja_binaries.py") .arg("--dir") .arg(&out_dir) .status() .expect("gn_ninja_binaries.py download failed"); assert!(status.success()); } assert!(gn.exists()); assert!(ninja.exists()); env::set_var("GN", gn); env::set_var("NINJA", ninja); } fn print_link_flags() { println!("cargo:rustc-link-lib=static=rusty_v8"); if cfg!(target_os = "windows") { println!("cargo:rustc-link-lib=dylib=winmm"); println!("cargo:rustc-link-lib=dylib=dbghelp"); } } fn need_gn_ninja_download() -> bool { !((which("ninja").is_ok() || env::var_os("NINJA").is_some()) && env::var_os("GN").is_some()) } // Download chromium's clang into OUT_DIR because Cargo will not allow us to // modify the source directory. fn clang_download() -> PathBuf { // TODO(ry) We can support clang 10 and above.... if that is installed use // it. /* if let Ok(clang_path) = which("clang") { // let bin_path = clang_path.parent().unwrap(); return bin_path.parent().unwrap().to_path_buf(); } */ let root = env::current_dir().unwrap(); let out_dir = env::var_os("OUT_DIR").unwrap(); let clang_base_path = root.join(out_dir).join("clang"); println!("clang_base_path {}", clang_base_path.display()); let status = Command::new("python") .arg("./tools/clang/scripts/update.py") .arg("--output-dir") .arg(&clang_base_path) .status() .expect("clang download failed"); assert!(status.success()); assert!(clang_base_path.exists()); clang_base_path } fn cc_wrapper(gn_args: &mut Vec, sccache_path: &Path) { gn_args.push(format!("cc_wrapper={:?}", sccache_path)); // Disable treat_warnings_as_errors until this sccache bug is fixed: // https://github.com/mozilla/sccache/issues/264 if cfg!(target_os = "windows") { gn_args.push("treat_warnings_as_errors=false".to_string()); } }