2015-11-19 17:20:12 -06:00
|
|
|
// Copyright 2015 The Rust Project Developers. See the COPYRIGHT
|
|
|
|
// file at the top-level directory of this distribution and at
|
|
|
|
// http://rust-lang.org/COPYRIGHT.
|
|
|
|
//
|
|
|
|
// Licensed under the Apache License, Version 2.0 <LICENSE-APACHE or
|
|
|
|
// http://www.apache.org/licenses/LICENSE-2.0> or the MIT license
|
|
|
|
// <LICENSE-MIT or http://opensource.org/licenses/MIT>, at your
|
|
|
|
// option. This file may not be copied, modified, or distributed
|
|
|
|
// except according to those terms.
|
|
|
|
|
|
|
|
use std::path::Path;
|
|
|
|
use std::process::Command;
|
|
|
|
use std::fs;
|
|
|
|
|
|
|
|
use build_helper::output;
|
|
|
|
use cmake;
|
2016-04-05 13:34:23 -05:00
|
|
|
use gcc;
|
2015-11-19 17:20:12 -06:00
|
|
|
|
|
|
|
use build::Build;
|
2016-04-05 13:34:23 -05:00
|
|
|
use build::util::{exe, staticlib, up_to_date};
|
2015-11-19 17:20:12 -06:00
|
|
|
|
|
|
|
pub fn llvm(build: &Build, target: &str) {
|
|
|
|
// If we're using a custom LLVM bail out here, but we can only use a
|
|
|
|
// custom LLVM for the build triple.
|
|
|
|
if let Some(config) = build.config.target_config.get(target) {
|
|
|
|
if let Some(ref s) = config.llvm_config {
|
|
|
|
return check_llvm_version(build, s);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the cleaning trigger is newer than our built artifacts (or if the
|
|
|
|
// artifacts are missing) then we keep going, otherwise we bail out.
|
|
|
|
let dst = build.llvm_out(target);
|
|
|
|
let stamp = build.src.join("src/rustllvm/llvm-auto-clean-trigger");
|
|
|
|
let llvm_config = dst.join("bin").join(exe("llvm-config", target));
|
|
|
|
build.clear_if_dirty(&dst, &stamp);
|
|
|
|
if fs::metadata(llvm_config).is_ok() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let _ = fs::remove_dir_all(&dst.join("build"));
|
|
|
|
t!(fs::create_dir_all(&dst.join("build")));
|
2016-03-26 06:35:45 -05:00
|
|
|
let assertions = if build.config.llvm_assertions {"ON"} else {"OFF"};
|
2015-11-19 17:20:12 -06:00
|
|
|
|
|
|
|
// http://llvm.org/docs/CMake.html
|
|
|
|
let mut cfg = cmake::Config::new(build.src.join("src/llvm"));
|
2016-04-09 23:27:32 -05:00
|
|
|
if build.config.ninja {
|
|
|
|
cfg.generator("Ninja");
|
|
|
|
}
|
2015-11-19 17:20:12 -06:00
|
|
|
cfg.target(target)
|
|
|
|
.host(&build.config.build)
|
|
|
|
.out_dir(&dst)
|
|
|
|
.profile(if build.config.llvm_optimize {"Release"} else {"Debug"})
|
|
|
|
.define("LLVM_ENABLE_ASSERTIONS", assertions)
|
|
|
|
.define("LLVM_TARGETS_TO_BUILD", "X86;ARM;AArch64;Mips;PowerPC")
|
|
|
|
.define("LLVM_INCLUDE_EXAMPLES", "OFF")
|
|
|
|
.define("LLVM_INCLUDE_TESTS", "OFF")
|
|
|
|
.define("LLVM_INCLUDE_DOCS", "OFF")
|
|
|
|
.define("LLVM_ENABLE_ZLIB", "OFF")
|
|
|
|
.define("WITH_POLLY", "OFF")
|
|
|
|
.define("LLVM_ENABLE_TERMINFO", "OFF")
|
|
|
|
.define("LLVM_ENABLE_LIBEDIT", "OFF")
|
|
|
|
.define("LLVM_PARALLEL_COMPILE_JOBS", build.jobs().to_string());
|
|
|
|
|
|
|
|
if target.starts_with("i686") {
|
|
|
|
cfg.define("LLVM_BUILD_32_BITS", "ON");
|
|
|
|
}
|
|
|
|
|
|
|
|
// http://llvm.org/docs/HowToCrossCompileLLVM.html
|
|
|
|
if target != build.config.build {
|
|
|
|
// FIXME: if the llvm root for the build triple is overridden then we
|
|
|
|
// should use llvm-tblgen from there, also should verify that it
|
|
|
|
// actually exists most of the time in normal installs of LLVM.
|
|
|
|
let host = build.llvm_out(&build.config.build).join("bin/llvm-tblgen");
|
|
|
|
cfg.define("CMAKE_CROSSCOMPILING", "True")
|
|
|
|
.define("LLVM_TARGET_ARCH", target.split('-').next().unwrap())
|
|
|
|
.define("LLVM_TABLEGEN", &host)
|
|
|
|
.define("LLVM_DEFAULT_TARGET_TRIPLE", target);
|
|
|
|
}
|
|
|
|
|
|
|
|
// MSVC handles compiler business itself
|
|
|
|
if !target.contains("msvc") {
|
|
|
|
if build.config.ccache {
|
|
|
|
cfg.define("CMAKE_C_COMPILER", "ccache")
|
|
|
|
.define("CMAKE_C_COMPILER_ARG1", build.cc(target))
|
|
|
|
.define("CMAKE_CXX_COMPILER", "ccache")
|
|
|
|
.define("CMAKE_CXX_COMPILER_ARG1", build.cxx(target));
|
|
|
|
} else {
|
|
|
|
cfg.define("CMAKE_C_COMPILER", build.cc(target))
|
|
|
|
.define("CMAKE_CXX_COMPILER", build.cxx(target));
|
|
|
|
}
|
|
|
|
cfg.build_arg("-j").build_arg(build.jobs().to_string());
|
2016-03-29 23:51:00 -05:00
|
|
|
|
|
|
|
cfg.define("CMAKE_C_FLAGS", build.cflags(target).join(" "));
|
|
|
|
cfg.define("CMAKE_CXX_FLAGS", build.cflags(target).join(" "));
|
2015-11-19 17:20:12 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: we don't actually need to build all LLVM tools and all LLVM
|
|
|
|
// libraries here, e.g. we just want a few components and a few
|
|
|
|
// tools. Figure out how to filter them down and only build the right
|
|
|
|
// tools and libs on all platforms.
|
|
|
|
cfg.build();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn check_llvm_version(build: &Build, llvm_config: &Path) {
|
|
|
|
if !build.config.llvm_version_check {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
let mut cmd = Command::new(llvm_config);
|
|
|
|
let version = output(cmd.arg("--version"));
|
|
|
|
if version.starts_with("3.5") || version.starts_with("3.6") ||
|
|
|
|
version.starts_with("3.7") {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
panic!("\n\nbad LLVM version: {}, need >=3.5\n\n", version)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn compiler_rt(build: &Build, target: &str) {
|
|
|
|
let dst = build.compiler_rt_out(target);
|
|
|
|
let arch = target.split('-').next().unwrap();
|
|
|
|
let mode = if build.config.rust_optimize {"Release"} else {"Debug"};
|
rustbuild: Fix cross compiling to FreeBSD
This commit fixes our support for cross compiling a compiler to run on FreeBSD.
Over the weekend I managed to get a cross compiler from Linux to FreeBSD [1]
which I hope to soon use to start producing FreeBSD nightly compilers. With the
`make dist` support added in #32237 we should be able to produce standard
rustc/rust-std packages for FreeBSD through a new slave with this cross compiler.
Currently, however, we don't "Just Work" when cross compiling FreeBSD and a
number of changes were required (part of this PR). They include:
* A few build fixes were needed in LLVM. Our own branch has been rebased on the
actual 3.8 release and I applied one extra commit [2] which contains two fixes:
1. The LLVM CMake build system passes the `-Wl,-z,defs` flag on many
platforms, but *not* when `CMAKE_SYSTEM_NAME` is "FreeBSD". Unfortunately
this doesn't take into account when we're cross compiling, and as predicted
the build will fail if `-Wl,-z,defs` is passed (see [3] for more info). To
fix this we test `TARGET_TRIPLE` instead of the `CMAKE_SYSTEM_NAME` which
is what we're compiling for which fixes the problem.
2. The `PATH_MAX` constant is apparently defined in a different location than
many other Unix systems, so a file which required this just needed some
help to keep compiling.
* Support for compiling compiler-rt with CMake has been added to rustbuild. It
looks like it just emulates Linux in what it compiles as it didn't seem to
naturally produce anything else... At least the architecture is right, so
seems good for now at least!
[1]: https://github.com/alexcrichton/port-of-rust/blob/master/prebuilt/freebsd/Dockerfile
[2]: https://github.com/rust-lang/llvm/commit/be89e4b5
[3]: https://bugs.webkit.org/show_bug.cgi?id=138420
2016-03-14 00:20:51 -05:00
|
|
|
let (dir, build_target, libname) = if target.contains("linux") ||
|
2016-03-17 15:58:26 -05:00
|
|
|
target.contains("freebsd") ||
|
|
|
|
target.contains("netbsd") {
|
2015-11-19 17:20:12 -06:00
|
|
|
let os = if target.contains("android") {"-android"} else {""};
|
2016-02-24 16:31:18 -06:00
|
|
|
let arch = if arch.starts_with("arm") && target.contains("eabihf") {
|
|
|
|
"armhf"
|
|
|
|
} else {
|
|
|
|
arch
|
|
|
|
};
|
2015-11-19 17:20:12 -06:00
|
|
|
let target = format!("clang_rt.builtins-{}{}", arch, os);
|
|
|
|
("linux".to_string(), target.clone(), target)
|
|
|
|
} else if target.contains("darwin") {
|
|
|
|
let target = format!("clang_rt.builtins_{}_osx", arch);
|
|
|
|
("builtins".to_string(), target.clone(), target)
|
|
|
|
} else if target.contains("windows-gnu") {
|
|
|
|
let target = format!("clang_rt.builtins-{}", arch);
|
|
|
|
("windows".to_string(), target.clone(), target)
|
|
|
|
} else if target.contains("windows-msvc") {
|
|
|
|
(format!("windows/{}", mode),
|
|
|
|
"lib/builtins/builtins".to_string(),
|
|
|
|
format!("clang_rt.builtins-{}", arch.replace("i686", "i386")))
|
|
|
|
} else {
|
|
|
|
panic!("can't get os from target: {}", target)
|
|
|
|
};
|
|
|
|
let output = dst.join("build/lib").join(dir)
|
|
|
|
.join(staticlib(&libname, target));
|
|
|
|
build.compiler_rt_built.borrow_mut().insert(target.to_string(),
|
|
|
|
output.clone());
|
|
|
|
if fs::metadata(&output).is_ok() {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
let _ = fs::remove_dir_all(&dst);
|
|
|
|
t!(fs::create_dir_all(&dst));
|
2016-04-05 13:34:23 -05:00
|
|
|
let build_llvm_config = build.llvm_config(&build.config.build);
|
2015-11-19 17:20:12 -06:00
|
|
|
let mut cfg = cmake::Config::new(build.src.join("src/compiler-rt"));
|
|
|
|
cfg.target(target)
|
|
|
|
.host(&build.config.build)
|
|
|
|
.out_dir(&dst)
|
|
|
|
.profile(mode)
|
|
|
|
.define("LLVM_CONFIG_PATH", build_llvm_config)
|
|
|
|
.define("COMPILER_RT_DEFAULT_TARGET_TRIPLE", target)
|
|
|
|
.define("COMPILER_RT_BUILD_SANITIZERS", "OFF")
|
|
|
|
.define("COMPILER_RT_BUILD_EMUTLS", "OFF")
|
2016-02-24 16:31:18 -06:00
|
|
|
// inform about c/c++ compilers, the c++ compiler isn't actually used but
|
|
|
|
// it's needed to get the initial configure to work on all platforms.
|
2015-11-19 17:20:12 -06:00
|
|
|
.define("CMAKE_C_COMPILER", build.cc(target))
|
2016-02-24 16:31:18 -06:00
|
|
|
.define("CMAKE_CXX_COMPILER", build.cc(target))
|
2015-11-19 17:20:12 -06:00
|
|
|
.build_target(&build_target);
|
|
|
|
cfg.build();
|
|
|
|
}
|
2016-04-05 13:34:23 -05:00
|
|
|
|
|
|
|
pub fn test_helpers(build: &Build, target: &str) {
|
|
|
|
let dst = build.test_helpers_out(target);
|
|
|
|
let src = build.src.join("src/rt/rust_test_helpers.c");
|
|
|
|
if up_to_date(&src, &dst.join("librust_test_helpers.a")) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
println!("Building test helpers");
|
|
|
|
t!(fs::create_dir_all(&dst));
|
|
|
|
let mut cfg = gcc::Config::new();
|
|
|
|
cfg.cargo_metadata(false)
|
|
|
|
.out_dir(&dst)
|
|
|
|
.target(target)
|
|
|
|
.host(&build.config.build)
|
|
|
|
.opt_level(0)
|
|
|
|
.debug(false)
|
|
|
|
.file(build.src.join("src/rt/rust_test_helpers.c"))
|
|
|
|
.compile("librust_test_helpers.a");
|
|
|
|
}
|