#![doc(html_root_url = "http://alexcrichton.com/gcc-rs")]
#![cfg_attr(test, deny(warnings))]
use std::env;
use std::ffi::OsString;
use std::fs;
use std::io;
use std::path::{PathBuf, Path};
use std::process::{Command, Stdio};
pub struct Config {
include_directories: Vec<PathBuf>,
definitions: Vec<(String, Option<String>)>,
objects: Vec<PathBuf>,
flags: Vec<String>,
files: Vec<PathBuf>,
cpp: bool,
cpp_link_stdlib: Option<String>,
cpp_set_stdlib: Option<String>,
target: String,
}
fn target_default_cpp_stdlib() -> Option<&'static str> {
let target = getenv_unwrap("TARGET");
if target.contains("msvc") {
None
} else if target.contains("darwin") {
Some("c++")
} else {
Some("stdc++")
}
}
fn getenv(v: &str) -> Option<String> {
let r = env::var(v).ok();
println!("{} = {:?}", v, r);
r
}
fn getenv_unwrap(v: &str) -> String {
match getenv(v) {
Some(s) => s,
None => fail(&format!("environment variable `{}` not defined", v)),
}
}
pub fn compile_library(output: &str, files: &[&str]) {
let mut c = Config::new();
for f in files.iter() {
c.file(*f);
}
c.compile(output)
}
impl Config {
pub fn new() -> Config {
Config {
include_directories: Vec::new(),
definitions: Vec::new(),
objects: Vec::new(),
flags: Vec::new(),
files: Vec::new(),
cpp: false,
cpp_link_stdlib: target_default_cpp_stdlib().map(|s| s.into()),
cpp_set_stdlib: None,
target: getenv_unwrap("TARGET"),
}
}
pub fn include<P: AsRef<Path>>(&mut self, dir: P) -> &mut Config {
self.include_directories.push(dir.as_ref().to_path_buf());
self
}
pub fn define(&mut self, var: &str, val: Option<&str>) -> &mut Config {
self.definitions.push((var.to_string(), val.map(|s| s.to_string())));
self
}
pub fn object<P: AsRef<Path>>(&mut self, obj: P) -> &mut Config {
self.objects.push(obj.as_ref().to_path_buf());
self
}
pub fn flag(&mut self, flag: &str) -> &mut Config {
self.flags.push(flag.to_string());
self
}
pub fn file<P: AsRef<Path>>(&mut self, p: P) -> &mut Config {
self.files.push(p.as_ref().to_path_buf());
self
}
pub fn cpp(&mut self, cpp: bool) -> &mut Config {
self.cpp = cpp;
self
}
pub fn cpp_link_stdlib(&mut self, cpp_link_stdlib: Option<&str>)
-> &mut Config {
self.cpp_link_stdlib = cpp_link_stdlib.map(|s| s.into());
self
}
pub fn cpp_set_stdlib(&mut self, cpp_set_stdlib: Option<&str>)
-> &mut Config {
self.cpp_set_stdlib = cpp_set_stdlib.map(|s| s.into());
self.cpp_link_stdlib(cpp_set_stdlib);
self
}
pub fn compile(&self, output: &str) {
assert!(output.starts_with("lib"));
assert!(output.ends_with(".a"));
let lib_name = &output[3..output.len() - 2];
let src = PathBuf::from(getenv_unwrap("CARGO_MANIFEST_DIR"));
let dst = PathBuf::from(getenv_unwrap("OUT_DIR"));
let mut objects = Vec::new();
for file in self.files.iter() {
let obj = dst.join(file).with_extension("o");
self.compile_object(&src.join(file), &obj);
objects.push(obj);
}
self.assemble(lib_name, &dst.join(output), &objects);
println!("cargo:rustc-link-search=native={}", dst.display());
println!("cargo:rustc-link-lib=static={}",
&output[3..output.len() - 2]);
if self.cpp {
if let Some(ref stdlib) = self.cpp_link_stdlib {
println!("cargo:rustc-link-lib={}", stdlib);
}
}
}
fn compile_object(&self, file: &Path, dst: &Path) {
let is_asm = file.extension().and_then(|s| s.to_str()) == Some("asm");
let msvc = self.target.contains("msvc");
let mut cmd = if msvc && is_asm {
self.msvc_macro_assembler()
} else {
self.compile_cmd()
};
if msvc {
cmd.arg("/nologo");
}
fs::create_dir_all(&dst.parent().unwrap()).unwrap();
if msvc && is_asm {
cmd.arg("/Fo").arg(dst);
} else if msvc {
let mut s = OsString::from("/Fo:");
s.push(&dst);
cmd.arg(s);
} else {
cmd.arg("-o").arg(&dst);
}
if msvc {
cmd.arg("/c");
}
cmd.arg(file);
run(&mut cmd, &self.compiler());
}
fn compiler(&self) -> String {
let (env, msvc, gnu, default) = if self.cpp {
("CXX", "cl", "g++", "c++")
} else {
("CC", "cl", "gcc", "cc")
};
get_var(env).unwrap_or(if cfg!(windows) {
if self.target.contains("msvc") {
msvc.to_string()
} else {
gnu.to_string()
}
} else if self.target.contains("android") {
format!("{}-{}", self.target, gnu)
} else {
default.to_string()
})
}
fn compile_cmd(&self) -> Command {
let opt_level = getenv_unwrap("OPT_LEVEL");
let profile = getenv_unwrap("PROFILE");
let msvc = self.target.contains("msvc");
println!("{} {}", profile, opt_level);
let mut cmd = Command::new(self.compiler());
if msvc {
cmd.arg("/MD"); cmd.arg(format!("/O{}", opt_level));
} else {
cmd.arg(format!("-O{}", opt_level));
cmd.arg("-c");
cmd.arg("-ffunction-sections").arg("-fdata-sections");
}
cmd.args(&envflags(if self.cpp {"CXXFLAGS"} else {"CFLAGS"}));
if profile == "debug" {
cmd.arg(if msvc {"/Z7"} else {"-g"});
}
if self.target.contains("-ios") {
self.ios_flags(&mut cmd);
} else if !msvc {
if self.target.contains("windows") {
cmd.arg("-mwin32");
}
if self.target.contains("i686") {
cmd.arg("-m32");
} else if self.target.contains("x86_64") {
cmd.arg("-m64");
}
if !self.target.contains("i686") {
cmd.arg("-fPIC");
}
}
if self.cpp && !msvc {
if let Some(ref stdlib) = self.cpp_set_stdlib {
cmd.arg(&format!("-stdlib=lib{}", stdlib));
}
}
for directory in self.include_directories.iter() {
cmd.arg(if msvc {"/I"} else {"-I"});
cmd.arg(directory);
}
for flag in self.flags.iter() {
cmd.arg(flag);
}
for &(ref key, ref value) in self.definitions.iter() {
let lead = if msvc {"/"} else {"-"};
if let &Some(ref value) = value {
cmd.arg(&format!("{}D{}={}", lead, key, value));
} else {
cmd.arg(&format!("{}D{}", lead, key));
}
}
return cmd
}
fn msvc_macro_assembler(&self) -> Command {
let mut cmd = if self.target.contains("x86_64") {
Command::new("ml64.exe")
} else {
Command::new("ml.exe")
};
for directory in self.include_directories.iter() {
cmd.arg("/I").arg(directory);
}
for &(ref key, ref value) in self.definitions.iter() {
if let &Some(ref value) = value {
cmd.arg(&format!("/D{}={}", key, value));
} else {
cmd.arg(&format!("/D{}", key));
}
}
return cmd
}
fn assemble(&self, lib_name: &str, dst: &Path, objects: &[PathBuf]) {
if self.target.contains("msvc") {
let mut out = OsString::from("/OUT:");
out.push(dst);
run(Command::new("lib.exe").arg(out).arg("/nologo")
.args(objects)
.args(&self.objects), "lib.exe");
let lib_dst = dst.with_file_name(format!("{}.lib", lib_name));
let _ = fs::remove_file(&lib_dst);
fs::hard_link(dst, lib_dst).unwrap();
} else {
let ar = get_var("AR").unwrap_or(if self.target.contains("android") {
format!("{}-ar", self.target)
} else {
"ar".to_string()
});
run(Command::new(&ar).arg("crus")
.arg(dst)
.args(objects)
.args(&self.objects), &ar);
}
}
fn ios_flags(&self, cmd: &mut Command) {
enum ArchSpec {
Device(&'static str),
Simulator(&'static str),
}
let arch = self.target.split('-').nth(0).unwrap();
let arch = match arch {
"arm" | "armv7" | "thumbv7" => ArchSpec::Device("armv7"),
"armv7s" | "thumbv7s" => ArchSpec::Device("armv7s"),
"arm64" | "aarch64" => ArchSpec::Device("arm64"),
"i386" | "i686" => ArchSpec::Simulator("-m32"),
"x86_64" => ArchSpec::Simulator("-m64"),
_ => fail("Unknown arch for iOS target")
};
let sdk = match arch {
ArchSpec::Device(arch) => {
cmd.arg("-arch").arg(arch);
"iphoneos"
},
ArchSpec::Simulator(arch) => {
cmd.arg(arch);
"iphonesimulator"
}
};
println!("Detecting iOS SDK path for {}", sdk);
let sdk_path = Command::new("xcrun")
.arg("--show-sdk-path")
.arg("--sdk")
.arg(sdk)
.stderr(Stdio::inherit())
.output()
.unwrap()
.stdout;
let sdk_path = String::from_utf8(sdk_path).unwrap();
cmd.arg("-isysroot");
cmd.arg(sdk_path.trim());
}
}
fn run(cmd: &mut Command, program: &str) {
println!("running: {:?}", cmd);
let status = match cmd.status() {
Ok(status) => status,
Err(ref e) if e.kind() == io::ErrorKind::NotFound => {
let extra = if cfg!(windows) {
" (see https://github.com/alexcrichton/gcc-rs#windows-notes \
for help)"
} else {
""
};
fail(&format!("failed to execute command: {}\nIs `{}` \
not installed?{}", e, program, extra));
}
Err(e) => fail(&format!("failed to execute command: {}", e)),
};
if !status.success() {
fail(&format!("command did not execute successfully, got: {}", status));
}
}
fn get_var(var_base: &str) -> Result<String, String> {
let target = getenv_unwrap("TARGET");
let host = getenv_unwrap("HOST");
let kind = if host == target {"HOST"} else {"TARGET"};
let target_u = target.replace("-", "_");
let res = getenv(&format!("{}_{}", var_base, target))
.or_else(|| getenv(&format!("{}_{}", var_base, target_u)))
.or_else(|| getenv(&format!("{}_{}", kind, var_base)))
.or_else(|| getenv(var_base));
match res {
Some(res) => Ok(res),
None => Err("Could not get environment variable".to_string()),
}
}
fn envflags(name: &str) -> Vec<String> {
get_var(name).unwrap_or(String::new())
.split(|c: char| c.is_whitespace()).filter(|s| !s.is_empty())
.map(|s| s.to_string())
.collect()
}
fn fail(s: &str) -> ! {
println!("\n\n{}\n\n", s);
panic!()
}