|
@ -46,6 +46,7 @@ |
|
|
#![cfg_attr(test, deny(warnings))] |
|
|
#![cfg_attr(test, deny(warnings))] |
|
|
|
|
|
|
|
|
use std::env; |
|
|
use std::env; |
|
|
|
|
|
use std::ffi::OsString; |
|
|
use std::fs; |
|
|
use std::fs; |
|
|
use std::io; |
|
|
use std::io; |
|
|
use std::path::{PathBuf, Path}; |
|
|
use std::path::{PathBuf, Path}; |
|
@ -189,19 +190,34 @@ impl Config { |
|
|
let dst = PathBuf::from(getenv_unwrap("OUT_DIR")); |
|
|
let dst = PathBuf::from(getenv_unwrap("OUT_DIR")); |
|
|
let mut objects = Vec::new(); |
|
|
let mut objects = Vec::new(); |
|
|
for file in self.files.iter() { |
|
|
for file in self.files.iter() { |
|
|
let mut cmd = self.compile_cmd(); |
|
|
let mut cmd = self.compile_cmd(&target); |
|
|
|
|
|
cmd.arg(src.join(file)); |
|
|
|
|
|
|
|
|
let obj = dst.join(file).with_extension("o"); |
|
|
let obj = dst.join(file).with_extension("o"); |
|
|
fs::create_dir_all(&obj.parent().unwrap()).unwrap(); |
|
|
fs::create_dir_all(&obj.parent().unwrap()).unwrap(); |
|
|
run(cmd.arg(&src.join(file)).arg("-o").arg(&obj), |
|
|
if target.contains("msvc") { |
|
|
&self.compiler(&target)); |
|
|
let mut s = OsString::from("/Fo:"); |
|
|
|
|
|
s.push(&obj); |
|
|
|
|
|
cmd.arg(s); |
|
|
|
|
|
} else { |
|
|
|
|
|
cmd.arg("-o").arg(&obj); |
|
|
|
|
|
} |
|
|
|
|
|
run(&mut cmd, &self.compiler(&target)); |
|
|
objects.push(obj); |
|
|
objects.push(obj); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
run(Command::new(&ar(&target)).arg("crus") |
|
|
if target.contains("msvc") { |
|
|
.arg(&dst.join(output)) |
|
|
let mut out = OsString::from("/OUT:"); |
|
|
.args(&objects) |
|
|
out.push(dst.join(output)); |
|
|
.args(&self.objects), |
|
|
run(Command::new("lib").arg(out).args(&objects).args(&self.objects), |
|
|
&ar(&target)); |
|
|
"lib"); |
|
|
|
|
|
} else { |
|
|
|
|
|
run(Command::new(&ar(&target)).arg("crus") |
|
|
|
|
|
.arg(&dst.join(output)) |
|
|
|
|
|
.args(&objects) |
|
|
|
|
|
.args(&self.objects), |
|
|
|
|
|
&ar(&target)); |
|
|
|
|
|
} |
|
|
println!("cargo:rustc-link-search=native={}", dst.display()); |
|
|
println!("cargo:rustc-link-search=native={}", dst.display()); |
|
|
println!("cargo:rustc-link-lib=static={}", |
|
|
println!("cargo:rustc-link-lib=static={}", |
|
|
&output[3..output.len() - 2]); |
|
|
&output[3..output.len() - 2]); |
|
@ -230,22 +246,26 @@ impl Config { |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
fn compile_cmd(&self) -> Command { |
|
|
fn compile_cmd(&self, target: &str) -> Command { |
|
|
let target = getenv_unwrap("TARGET"); |
|
|
|
|
|
let opt_level = getenv_unwrap("OPT_LEVEL"); |
|
|
let opt_level = getenv_unwrap("OPT_LEVEL"); |
|
|
let profile = getenv_unwrap("PROFILE"); |
|
|
let profile = getenv_unwrap("PROFILE"); |
|
|
println!("{} {}", profile, opt_level); |
|
|
println!("{} {}", profile, opt_level); |
|
|
|
|
|
|
|
|
let mut cmd = Command::new(self.compiler(&target)); |
|
|
let mut cmd = Command::new(self.compiler(&target)); |
|
|
|
|
|
|
|
|
cmd.arg(format!("-O{}", opt_level)); |
|
|
if target.contains("msvc") { |
|
|
cmd.arg("-c"); |
|
|
cmd.arg("/c"); |
|
|
cmd.arg("-ffunction-sections").arg("-fdata-sections"); |
|
|
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(&self.compile_flags()); |
|
|
cmd.args(&self.compile_flags()); |
|
|
|
|
|
|
|
|
if target.contains("-ios") { |
|
|
if target.contains("-ios") { |
|
|
cmd.args(&ios_flags(&target)); |
|
|
cmd.args(&ios_flags(&target)); |
|
|
} else { |
|
|
} else if !target.contains("msvc") { |
|
|
if target.contains("windows") { |
|
|
if target.contains("windows") { |
|
|
cmd.arg("-mwin32"); |
|
|
cmd.arg("-mwin32"); |
|
|
} |
|
|
} |
|
@ -268,7 +288,8 @@ impl Config { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
for directory in self.include_directories.iter() { |
|
|
for directory in self.include_directories.iter() { |
|
|
cmd.arg("-I").arg(directory); |
|
|
cmd.arg(if target.contains("msvc") {"/I"} else {"-I"}); |
|
|
|
|
|
cmd.arg(directory); |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
for flag in self.flags.iter() { |
|
|
for flag in self.flags.iter() { |
|
@ -276,10 +297,11 @@ impl Config { |
|
|
} |
|
|
} |
|
|
|
|
|
|
|
|
for &(ref key, ref value) in self.definitions.iter() { |
|
|
for &(ref key, ref value) in self.definitions.iter() { |
|
|
|
|
|
let lead = if target.contains("msvc") {"/"} else {"-"}; |
|
|
if let &Some(ref value) = value { |
|
|
if let &Some(ref value) = value { |
|
|
cmd.arg(&format!("-D{}={}", key, value)); |
|
|
cmd.arg(&format!("{}D{}={}", lead, key, value)); |
|
|
} else { |
|
|
} else { |
|
|
cmd.arg(&format!("-D{}", key)); |
|
|
cmd.arg(&format!("{}D{}", lead, key)); |
|
|
} |
|
|
} |
|
|
} |
|
|
} |
|
|
return cmd; |
|
|
return cmd; |
|
@ -291,12 +313,14 @@ fn run(cmd: &mut Command, program: &str) { |
|
|
let status = match cmd.status() { |
|
|
let status = match cmd.status() { |
|
|
Ok(status) => status, |
|
|
Ok(status) => status, |
|
|
Err(ref e) if e.kind() == io::ErrorKind::NotFound => { |
|
|
Err(ref e) if e.kind() == io::ErrorKind::NotFound => { |
|
|
fail(&format!("failed to execute command: {}\nIs `{}` not installed?{}", |
|
|
let extra = if cfg!(windows) { |
|
|
e, program, |
|
|
" (see https://github.com/alexcrichton/gcc-rs#windows-notes \
|
|
|
if cfg!(windows) { |
|
|
for help)" |
|
|
" (see https://github.com/alexcrichton/gcc-rs#windows-notes for help)" |
|
|
} else { |
|
|
} else {""} |
|
|
"" |
|
|
)); |
|
|
}; |
|
|
|
|
|
fail(&format!("failed to execute command: {}\nIs `{}` \ |
|
|
|
|
|
not installed?{}", e, program, extra)); |
|
|
} |
|
|
} |
|
|
Err(e) => fail(&format!("failed to execute command: {}", e)), |
|
|
Err(e) => fail(&format!("failed to execute command: {}", e)), |
|
|
}; |
|
|
}; |
|
@ -325,7 +349,11 @@ fn gcc(target: &str) -> String { |
|
|
let is_android = target.find("android").is_some(); |
|
|
let is_android = target.find("android").is_some(); |
|
|
|
|
|
|
|
|
get_var("CC").unwrap_or(if cfg!(windows) { |
|
|
get_var("CC").unwrap_or(if cfg!(windows) { |
|
|
"gcc".to_string() |
|
|
if target.contains("msvc") { |
|
|
|
|
|
"cl".to_string() |
|
|
|
|
|
} else { |
|
|
|
|
|
"gcc".to_string() |
|
|
|
|
|
} |
|
|
} else if is_android { |
|
|
} else if is_android { |
|
|
format!("{}-gcc", target) |
|
|
format!("{}-gcc", target) |
|
|
} else { |
|
|
} else { |
|
@ -337,7 +365,11 @@ fn gxx(target: &str) -> String { |
|
|
let is_android = target.find("android").is_some(); |
|
|
let is_android = target.find("android").is_some(); |
|
|
|
|
|
|
|
|
get_var("CXX").unwrap_or(if cfg!(windows) { |
|
|
get_var("CXX").unwrap_or(if cfg!(windows) { |
|
|
"g++".to_string() |
|
|
if target.contains("msvc") { |
|
|
|
|
|
"cl".to_string() |
|
|
|
|
|
} else { |
|
|
|
|
|
"g++".to_string() |
|
|
|
|
|
} |
|
|
} else if is_android { |
|
|
} else if is_android { |
|
|
format!("{}-g++", target) |
|
|
format!("{}-g++", target) |
|
|
} else { |
|
|
} else { |
|
|