#![cfg_attr(feature = "fail-on-warnings", deny(warnings))]
use std::cmp::Ordering;
use std::env;
use std::ffi::{OsStr, OsString};
use std::fs;
use std::fs::File;
use std::io::{BufRead, BufReader, BufWriter, Result as IoResult, Write};
#[cfg(unix)]
use std::os::unix::fs as unix_fs;
#[cfg(windows)]
use std::os::windows::fs as windows_fs;
use std::path::{Path, PathBuf};
use std::process::{Command, Stdio};
use std::str;
const GMP_DIR: &str = "gmp-6.3.0-c";
const MPFR_DIR: &str = "mpfr-4.2.1-c";
const MPC_DIR: &str = "mpc-1.3.1-c";
const GMP_VER: (i32, i32, i32) = (6, 3, 0);
const MPFR_VER: (i32, i32, i32) = (4, 2, 1);
const MPC_VER: (i32, i32, i32) = (1, 3, 1);
#[derive(Clone, Copy, PartialEq)]
enum Target {
Mingw,
Msvc,
Other,
}
struct Environment {
rustc: OsString,
c_compiler: OsString,
target: Target,
cross_target: Option<String>,
c_no_tests: bool,
src_dir: PathBuf,
out_dir: PathBuf,
lib_dir: PathBuf,
include_dir: PathBuf,
build_dir: PathBuf,
cache_dir: Option<PathBuf>,
jobs: OsString,
version_prefix: String,
version_patch: Option<u64>,
use_system_libs: bool,
}
fn main() {
let rustc = cargo_env("RUSTC");
let cc = env::var_os("CC");
let cc_cache_dir = cc.as_ref().map(|cc| {
let mut dir = OsString::from("CC-");
dir.push(cc);
dir
});
let c_compiler = cc.unwrap_or_else(|| "gcc".into());
let cflags = env::var_os("CFLAGS");
let cflags_cache_dir = cflags.as_ref().map(|cflags| {
#[allow(deprecated)]
{
use std::hash::{Hash, Hasher, SipHasher};
let mut hasher = SipHasher::new();
cflags.hash(&mut hasher);
let hash = hasher.finish();
OsString::from(format!("CFLAGS-{hash:016X}"))
}
});
let host = cargo_env("HOST")
.into_string()
.expect("env var HOST having sensible characters");
let raw_target = cargo_env("TARGET")
.into_string()
.expect("env var TARGET having sensible characters");
let force_cross = there_is_env("CARGO_FEATURE_FORCE_CROSS");
if !force_cross && !compilation_target_allowed(&host, &raw_target) {
panic!(
"Cross compilation from {host} to {raw_target} not supported! \
Use the `force-cross` feature to cross compile anyway."
);
}
let target = if raw_target.contains("-windows-msvc") {
Target::Msvc
} else if raw_target.contains("-windows-gnu") {
Target::Mingw
} else {
Target::Other
};
let cross_target = if host == raw_target {
None
} else {
Some(raw_target)
};
let c_no_tests = there_is_env("CARGO_FEATURE_C_NO_TESTS");
let src_dir = PathBuf::from(cargo_env("CARGO_MANIFEST_DIR"));
let out_dir = PathBuf::from(cargo_env("OUT_DIR"));
let (version_prefix, version_patch) = get_version();
println!("cargo:rerun-if-env-changed=GMP_MPFR_SYS_CACHE");
let cache_dir = match env::var_os("GMP_MPFR_SYS_CACHE") {
Some(ref c) if c.is_empty() || c == "_" => None,
Some(c) => Some(PathBuf::from(c)),
None => system_cache_dir().map(|c| c.join("gmp-mpfr-sys")),
};
let cache_target = cross_target.as_ref().unwrap_or(&host);
let cache_dir = cache_dir
.map(|cache| cache.join(&version_prefix))
.map(|cache| cache.join(cache_target))
.map(|cache| match cc_cache_dir {
Some(dir) => cache.join(dir),
None => cache,
})
.map(|cache| match cflags_cache_dir {
Some(dir) => cache.join(dir),
None => cache,
});
let use_system_libs = there_is_env("CARGO_FEATURE_USE_SYSTEM_LIBS");
if use_system_libs {
match target {
Target::Msvc => panic!("the use-system-libs feature is not supported on this target"),
Target::Mingw => mingw_pkg_config_libdir_or_panic(),
_ => {}
}
}
let env = Environment {
rustc,
c_compiler,
target,
cross_target,
c_no_tests,
src_dir,
out_dir: out_dir.clone(),
lib_dir: out_dir.join("lib"),
include_dir: out_dir.join("include"),
build_dir: out_dir.join("build"),
cache_dir,
jobs: cargo_env("NUM_JOBS"),
version_prefix,
version_patch,
use_system_libs,
};
create_dir_or_panic(&env.lib_dir);
create_dir_or_panic(&env.include_dir);
if env.use_system_libs {
check_system_libs(&env);
} else {
compile_libs(&env);
}
}
fn check_system_libs(env: &Environment) {
let build_dir_existed = env.build_dir.exists();
let try_dir = env.build_dir.join("system_libs");
remove_dir_or_panic(&try_dir);
create_dir_or_panic(&try_dir);
println!("$ cd {try_dir:?}");
println!("$ #Check for system GMP");
create_file_or_panic(&try_dir.join("system_gmp.c"), SYSTEM_GMP_C);
let mut cmd = Command::new(&env.c_compiler);
cmd.current_dir(&try_dir)
.args(["-fPIC", "system_gmp.c", "-lgmp", "-o", "system_gmp.exe"]);
execute(cmd);
cmd = Command::new(try_dir.join("system_gmp.exe"));
cmd.current_dir(&try_dir);
execute(cmd);
process_gmp_header(
env,
&try_dir.join("system_gmp.out"),
Some(&env.out_dir.join("gmp_h.rs")),
)
.unwrap_or_else(|e| panic!("{}", e));
let feature_mpfr = there_is_env("CARGO_FEATURE_MPFR");
let feature_mpc = there_is_env("CARGO_FEATURE_MPC");
if feature_mpfr {
println!("$ #Check for system MPFR");
create_file_or_panic(&try_dir.join("system_mpfr.c"), SYSTEM_MPFR_C);
cmd = Command::new(&env.c_compiler);
cmd.current_dir(&try_dir).args([
"-fPIC",
"system_mpfr.c",
"-lmpfr",
"-lgmp",
"-o",
"system_mpfr.exe",
]);
execute(cmd);
cmd = Command::new(try_dir.join("system_mpfr.exe"));
cmd.current_dir(&try_dir);
execute(cmd);
process_mpfr_header(
env,
&try_dir.join("system_mpfr.out"),
Some(&env.out_dir.join("mpfr_h.rs")),
)
.unwrap_or_else(|e| panic!("{}", e));
}
if feature_mpc {
println!("$ #Check for system MPC");
create_file_or_panic(&try_dir.join("system_mpc.c"), SYSTEM_MPC_C);
cmd = Command::new(&env.c_compiler);
cmd.current_dir(&try_dir).args([
"-fPIC",
"system_mpc.c",
"-lmpc",
"-lgmp",
"-o",
"system_mpc.exe",
]);
execute(cmd);
cmd = Command::new(try_dir.join("system_mpc.exe"));
cmd.current_dir(&try_dir);
execute(cmd);
process_mpc_header(
env,
&try_dir.join("system_mpc.out"),
Some(&env.out_dir.join("mpc_h.rs")),
)
.unwrap_or_else(|e| panic!("{}", e));
}
if !there_is_env("CARGO_FEATURE_CNODELETE") {
if build_dir_existed {
let _ = remove_dir(&try_dir);
} else {
remove_dir_or_panic(&env.build_dir);
}
}
write_link_info(env, feature_mpfr, feature_mpc);
}
fn compile_libs(env: &Environment) {
let gmp_ah = (env.lib_dir.join("libgmp.a"), env.include_dir.join("gmp.h"));
let mpc_ah = if there_is_env("CARGO_FEATURE_MPC") {
Some((env.lib_dir.join("libmpc.a"), env.include_dir.join("mpc.h")))
} else {
None
};
let mpfr_ah = if mpc_ah.is_some() || there_is_env("CARGO_FEATURE_MPFR") {
Some((
env.lib_dir.join("libmpfr.a"),
env.include_dir.join("mpfr.h"),
))
} else {
None
};
let NeedCompile {
gmp: compile_gmp,
mpfr: compile_mpfr,
mpc: compile_mpc,
} = need_compile(env, &gmp_ah, &mpfr_ah, &mpc_ah);
if compile_gmp {
check_for_msvc(env);
remove_dir_or_panic(&env.build_dir);
create_dir_or_panic(&env.build_dir);
link_dir(&env.src_dir.join(GMP_DIR), &env.build_dir.join("gmp-src"));
let (ref a, ref h) = gmp_ah;
build_gmp(env, a, h);
}
if compile_mpfr {
link_dir(&env.src_dir.join(MPFR_DIR), &env.build_dir.join("mpfr-src"));
let (ref a, ref h) = *mpfr_ah.as_ref().unwrap();
build_mpfr(env, a, h);
}
if compile_mpc {
link_dir(&env.src_dir.join(MPC_DIR), &env.build_dir.join("mpc-src"));
let (ref a, ref h) = *mpc_ah.as_ref().unwrap();
build_mpc(env, a, h);
}
if compile_gmp {
if !there_is_env("CARGO_FEATURE_CNODELETE") {
remove_dir_or_panic(&env.build_dir);
}
if save_cache(env, &gmp_ah, &mpfr_ah, &mpc_ah) {
clear_cache_redundancies(env, mpfr_ah.is_some(), mpc_ah.is_some());
}
}
process_gmp_header(env, &gmp_ah.1, Some(&env.out_dir.join("gmp_h.rs")))
.unwrap_or_else(|e| panic!("{}", e));
if let Some(ref mpfr_ah) = mpfr_ah {
process_mpfr_header(env, &mpfr_ah.1, Some(&env.out_dir.join("mpfr_h.rs")))
.unwrap_or_else(|e| panic!("{}", e));
}
if let Some(ref mpc_ah) = mpc_ah {
process_mpc_header(env, &mpc_ah.1, Some(&env.out_dir.join("mpc_h.rs")))
.unwrap_or_else(|e| panic!("{}", e));
}
write_link_info(env, mpfr_ah.is_some(), mpc_ah.is_some());
}
fn get_version() -> (String, Option<u64>) {
let version = cargo_env("CARGO_PKG_VERSION")
.into_string()
.unwrap_or_else(|e| panic!("version not in utf-8: {e:?}"));
let last_dot = version
.rfind('.')
.unwrap_or_else(|| panic!("version has no dots: {version}"));
if last_dot == 0 {
panic!("version starts with dot: {version}");
}
match version[last_dot + 1..].parse::<u64>() {
Ok(patch) => {
let mut v = version;
v.truncate(last_dot);
(v, Some(patch))
}
Err(_) => (version, None),
}
}
struct NeedCompile {
gmp: bool,
mpfr: bool,
mpc: bool,
}
fn need_compile(
env: &Environment,
gmp_ah: &(PathBuf, PathBuf),
mpfr_ah: &Option<(PathBuf, PathBuf)>,
mpc_ah: &Option<(PathBuf, PathBuf)>,
) -> NeedCompile {
let gmp_fine = gmp_ah.0.is_file() && gmp_ah.1.is_file();
let mpfr_fine = match *mpfr_ah {
Some((ref a, ref h)) => a.is_file() && h.is_file(),
None => true,
};
let mpc_fine = match *mpc_ah {
Some((ref a, ref h)) => a.is_file() && h.is_file(),
None => true,
};
if gmp_fine && mpfr_fine && mpc_fine {
if should_save_cache(env, mpfr_ah.is_some(), mpc_ah.is_some())
&& save_cache(env, gmp_ah, mpfr_ah, mpc_ah)
{
clear_cache_redundancies(env, mpfr_ah.is_some(), mpc_ah.is_some());
}
return NeedCompile {
gmp: false,
mpfr: false,
mpc: false,
};
} else if load_cache(env, gmp_ah, mpfr_ah, mpc_ah) {
return NeedCompile {
gmp: false,
mpfr: false,
mpc: false,
};
}
let need_mpc = !mpc_fine;
let need_mpfr = need_mpc || !mpfr_fine;
let need_gmp = need_mpfr || !gmp_fine;
NeedCompile {
gmp: need_gmp,
mpfr: need_mpfr,
mpc: need_mpc,
}
}
fn save_cache(
env: &Environment,
gmp_ah: &(PathBuf, PathBuf),
mpfr_ah: &Option<(PathBuf, PathBuf)>,
mpc_ah: &Option<(PathBuf, PathBuf)>,
) -> bool {
let cache_dir = match env.cache_dir {
Some(ref s) => s,
None => return false,
};
let version_dir = match env.version_patch {
None => cache_dir.join(&env.version_prefix),
Some(patch) => cache_dir.join(format!("{}.{}", env.version_prefix, patch)),
};
let mut ok = create_dir(&version_dir).is_ok();
let dir = if env.c_no_tests {
let no_tests_dir = version_dir.join("c-no-tests");
ok = ok && create_dir(&no_tests_dir).is_ok();
no_tests_dir
} else {
version_dir
};
let (ref a, ref h) = *gmp_ah;
ok = ok && copy_file(a, &dir.join("libgmp.a")).is_ok();
ok = ok && copy_file(h, &dir.join("gmp.h")).is_ok();
if let Some((ref a, ref h)) = *mpfr_ah {
ok = ok && copy_file(a, &dir.join("libmpfr.a")).is_ok();
ok = ok && copy_file(h, &dir.join("mpfr.h")).is_ok();
}
if let Some((ref a, ref h)) = *mpc_ah {
ok = ok && copy_file(a, &dir.join("libmpc.a")).is_ok();
ok = ok && copy_file(h, &dir.join("mpc.h")).is_ok();
}
ok
}
fn clear_cache_redundancies(env: &Environment, mpfr: bool, mpc: bool) {
let Some(cache_dir) = &env.cache_dir else {
return;
};
let cache_dirs = cache_directories(env, cache_dir)
.into_iter()
.rev()
.filter(|x| match env.version_patch {
None => x.1.is_none(),
Some(patch) => x.1.map(|p| p <= patch).unwrap_or(false),
});
for (version_dir, version_patch) in cache_dirs {
let no_tests_dir = version_dir.join("c-no-tests");
if version_patch == env.version_patch {
if !env.c_no_tests
&& (mpc || !no_tests_dir.join("libmpc.a").is_file())
&& (mpfr || !no_tests_dir.join("libmpfr.a").is_file())
{
let _ = remove_dir(&no_tests_dir);
}
continue;
}
if (!mpc && no_tests_dir.join("libmpc.a").is_file())
|| (!mpfr && no_tests_dir.join("libmpfr.a").is_file())
{
continue;
}
let _ = remove_dir(&no_tests_dir);
let delete_version_dir_condition = if env.c_no_tests {
!version_dir.join("libgmp.a").is_file()
} else {
(mpc || !version_dir.join("libmpc.a").is_file())
&& (mpfr || !version_dir.join("libmpfr.a").is_file())
};
if delete_version_dir_condition {
let _ = remove_dir(&version_dir);
}
}
}
fn cache_directories(env: &Environment, base: &Path) -> Vec<(PathBuf, Option<u64>)> {
let Ok(dir) = fs::read_dir(base) else {
return Vec::new();
};
let mut vec = Vec::new();
for entry in dir {
let Ok(e) = entry else { continue };
let path = e.path();
if !path.is_dir() {
continue;
}
let patch = {
let Some(file_name) = path.file_name() else {
continue;
};
let Some(path_str) = file_name.to_str() else {
continue;
};
if path_str == env.version_prefix {
None
} else if !path_str.starts_with(&env.version_prefix)
|| !path_str[env.version_prefix.len()..].starts_with('.')
{
continue;
} else {
match path_str[env.version_prefix.len() + 1..].parse::<u64>() {
Ok(patch) => Some(patch),
Err(_) => continue,
}
}
};
vec.push((path, patch));
}
vec.sort_by_key(|k| k.1);
vec
}
fn load_cache(
env: &Environment,
gmp_ah: &(PathBuf, PathBuf),
mpfr_ah: &Option<(PathBuf, PathBuf)>,
mpc_ah: &Option<(PathBuf, PathBuf)>,
) -> bool {
let Some(cache_dir) = &env.cache_dir else {
return false;
};
let env_version_patch = env.version_patch;
let cache_dirs = cache_directories(env, cache_dir)
.into_iter()
.rev()
.filter(|x| match env_version_patch {
None => x.1.is_none(),
Some(patch) => x.1.map(|p| p >= patch).unwrap_or(false),
})
.collect::<Vec<_>>();
let suffixes: &[Option<&str>] = if env.c_no_tests {
&[None, Some("c-no-tests")]
} else {
&[None]
};
for suffix in suffixes {
for (version_dir, _) in &cache_dirs {
let joined;
let dir = if let Some(suffix) = suffix {
joined = version_dir.join(suffix);
&joined
} else {
version_dir
};
let mut ok = true;
if let Some((ref a, ref h)) = *mpc_ah {
ok = ok && copy_file(&dir.join("libmpc.a"), a).is_ok();
let header = dir.join("mpc.h");
ok = ok && process_mpc_header(env, &header, None).is_ok();
ok = ok && copy_file(&header, h).is_ok();
}
if let Some((ref a, ref h)) = *mpfr_ah {
ok = ok && copy_file(&dir.join("libmpfr.a"), a).is_ok();
let header = dir.join("mpfr.h");
ok = ok && process_mpfr_header(env, &header, None).is_ok();
ok = ok && copy_file(&header, h).is_ok();
}
let (ref a, ref h) = *gmp_ah;
ok = ok && copy_file(&dir.join("libgmp.a"), a).is_ok();
let header = dir.join("gmp.h");
ok = ok && process_gmp_header(env, &header, None).is_ok();
ok = ok && copy_file(&header, h).is_ok();
if ok {
return true;
}
}
}
false
}
fn should_save_cache(env: &Environment, mpfr: bool, mpc: bool) -> bool {
let Some(cache_dir) = &env.cache_dir else {
return false;
};
let cache_dirs = cache_directories(env, cache_dir)
.into_iter()
.rev()
.filter(|x| match env.version_patch {
None => x.1.is_none(),
Some(patch) => x.1.map(|p| p >= patch).unwrap_or(false),
})
.collect::<Vec<_>>();
let suffixes: &[Option<&str>] = if env.c_no_tests {
&[None, Some("c-no-tests")]
} else {
&[None]
};
for suffix in suffixes {
for (version_dir, _) in &cache_dirs {
let joined;
let dir = if let Some(suffix) = suffix {
joined = version_dir.join(suffix);
&joined
} else {
version_dir
};
let mut ok = true;
if mpc {
ok = ok && dir.join("libmpc.a").is_file();
ok = ok && dir.join("mpc.h").is_file();
}
if mpfr {
ok = ok && dir.join("libmpfr.a").is_file();
ok = ok && dir.join("mpfr.h").is_file();
}
ok = ok && dir.join("libgmp.a").is_file();
ok = ok && dir.join("gmp.h").is_file();
if ok {
return false;
}
}
}
true
}
fn get_actual_cross_target(cross_target: &str) -> &str {
match cross_target {
"x86_64-pc-windows-gnu" => "x86_64-w64-mingw32",
_ => cross_target,
}
}
fn build_gmp(env: &Environment, lib: &Path, header: &Path) {
let build_dir = env.build_dir.join("gmp-build");
create_dir_or_panic(&build_dir);
println!("$ cd {build_dir:?}");
let mut conf = String::from("../gmp-src/configure --enable-fat --disable-shared --with-pic");
if let Some(cross_target) = env.cross_target.as_ref() {
conf.push_str(" --host ");
conf.push_str(get_actual_cross_target(cross_target));
}
configure(&build_dir, &OsString::from(conf));
make_and_check(env, &build_dir);
let build_lib = build_dir.join(".libs").join("libgmp.a");
copy_file_or_panic(&build_lib, lib);
let build_header = build_dir.join("gmp.h");
copy_file_or_panic(&build_header, header);
}
fn compatible_version(
env: &Environment,
major: i32,
minor: i32,
patchlevel: i32,
expected: (i32, i32, i32),
) -> bool {
(major == expected.0 && minor >= expected.1)
&& (minor > expected.1 || patchlevel >= expected.2 || env.use_system_libs)
}
fn process_gmp_header(
env: &Environment,
header: &Path,
out_file: Option<&Path>,
) -> Result<(), String> {
let mut major = None;
let mut minor = None;
let mut patchlevel = None;
let mut limb_bits = None;
let mut nail_bits = None;
let mut long_long_limb = None;
let mut cc = None;
let mut cflags = None;
let mut reader = open(header);
let mut buf = String::new();
while read_line(&mut reader, &mut buf, header) > 0 {
let s = "#define __GNU_MP_VERSION ";
if let Some(start) = buf.find(s) {
major = buf[(start + s.len())..].trim().parse::<i32>().ok();
}
let s = "#define __GNU_MP_VERSION_MINOR ";
if let Some(start) = buf.find(s) {
minor = buf[(start + s.len())..].trim().parse::<i32>().ok();
}
let s = "#define __GNU_MP_VERSION_PATCHLEVEL ";
if let Some(start) = buf.find(s) {
patchlevel = buf[(start + s.len())..].trim().parse::<i32>().ok();
}
if buf.contains("#undef _LONG_LONG_LIMB") {
long_long_limb = Some(false);
}
if buf.contains("#define _LONG_LONG_LIMB 1") {
long_long_limb = Some(true);
}
let s = "#define GMP_LIMB_BITS ";
if let Some(start) = buf.find(s) {
limb_bits = buf[(start + s.len())..].trim().parse::<i32>().ok();
}
let s = "#define GMP_NAIL_BITS ";
if let Some(start) = buf.find(s) {
nail_bits = buf[(start + s.len())..].trim().parse::<i32>().ok();
}
let s = "#define __GMP_CC ";
if let Some(start) = buf.find(s) {
cc = Some(
buf[(start + s.len())..]
.trim()
.trim_matches('"')
.to_string(),
);
}
let s = "#define __GMP_CFLAGS ";
if let Some(start) = buf.find(s) {
cflags = Some(
buf[(start + s.len())..]
.trim()
.trim_matches('"')
.to_string(),
);
}
buf.clear();
}
drop(reader);
let major = major.expect("Cannot determine __GNU_MP_VERSION");
let minor = minor.expect("Cannot determine __GNU_MP_VERSION_MINOR");
let patchlevel = patchlevel.expect("Cannot determine __GNU_MP_VERSION_PATCHLEVEL");
if !compatible_version(env, major, minor, patchlevel, GMP_VER) {
return Err(format!(
"This version of gmp-mpfr-sys supports GMP {}.{}.{}, but {}.{}.{} was found",
GMP_VER.0, GMP_VER.1, GMP_VER.2, major, minor, patchlevel
));
}
let limb_bits = limb_bits.expect("Cannot determine GMP_LIMB_BITS");
println!("cargo:limb_bits={limb_bits}");
let nail_bits = nail_bits.expect("Cannot determine GMP_NAIL_BITS");
if nail_bits > 0 {
println!("cargo:rustc-cfg=nails");
}
let long_long_limb = long_long_limb.expect("Cannot determine _LONG_LONG_LIMB");
let long_long_limb = if long_long_limb {
println!("cargo:rustc-cfg=long_long_limb");
"libc::c_ulonglong"
} else {
"c_ulong"
};
let cc = cc.expect("Cannot determine __GMP_CC");
let cflags = cflags.expect("Cannot determine __GMP_CFLAGS");
let content = format!(
concat!(
"const GMP_VERSION: c_int = {};\n",
"const GMP_VERSION_MINOR: c_int = {};\n",
"const GMP_VERSION_PATCHLEVEL: c_int = {};\n",
"const GMP_LIMB_BITS: c_int = {};\n",
"const GMP_NAIL_BITS: c_int = {};\n",
"type GMP_LIMB_T = {};\n",
"const GMP_CC: *const c_char = b\"{}\\0\".as_ptr().cast();\n",
"const GMP_CFLAGS: *const c_char = b\"{}\\0\".as_ptr().cast();\n"
),
major, minor, patchlevel, limb_bits, nail_bits, long_long_limb, cc, cflags
);
if let Some(out_file) = out_file {
let mut rs = create(out_file);
write_flush(&mut rs, &content, out_file);
}
Ok(())
}
fn process_mpfr_header(
env: &Environment,
header: &Path,
out_file: Option<&Path>,
) -> Result<(), String> {
let mut major = None;
let mut minor = None;
let mut patchlevel = None;
let mut version = None;
let mut reader = open(header);
let mut buf = String::new();
while read_line(&mut reader, &mut buf, header) > 0 {
let s = "#define MPFR_VERSION_MAJOR ";
if let Some(start) = buf.find(s) {
major = buf[(start + s.len())..].trim().parse::<i32>().ok();
}
let s = "#define MPFR_VERSION_MINOR ";
if let Some(start) = buf.find(s) {
minor = buf[(start + s.len())..].trim().parse::<i32>().ok();
}
let s = "#define MPFR_VERSION_PATCHLEVEL ";
if let Some(start) = buf.find(s) {
patchlevel = buf[(start + s.len())..].trim().parse::<i32>().ok();
}
let s = "#define MPFR_VERSION_STRING ";
if let Some(start) = buf.find(s) {
version = Some(
buf[(start + s.len())..]
.trim()
.trim_matches('"')
.to_string(),
);
}
buf.clear();
}
drop(reader);
let major = major.expect("Cannot determine MPFR_VERSION_MAJOR");
let minor = minor.expect("Cannot determine MPFR_VERSION_MINOR");
let patchlevel = patchlevel.expect("Cannot determine MPFR_VERSION_PATCHLEVEL");
if !compatible_version(env, major, minor, patchlevel, MPFR_VER) {
return Err(format!(
"This version of gmp-mpfr-sys supports MPFR {}.{}.{}, but {}.{}.{} was found",
MPFR_VER.0, MPFR_VER.1, MPFR_VER.2, major, minor, patchlevel
));
}
let version = version.expect("Cannot determine MPFR_VERSION_STRING");
let content = format!(
concat!(
"const MPFR_VERSION_MAJOR: c_int = {};\n",
"const MPFR_VERSION_MINOR: c_int = {};\n",
"const MPFR_VERSION_PATCHLEVEL: c_int = {};\n",
"const MPFR_VERSION_STRING: *const c_char = b\"{}\\0\".as_ptr().cast();\n"
),
major, minor, patchlevel, version
);
if let Some(out_file) = out_file {
let mut rs = create(out_file);
write_flush(&mut rs, &content, out_file);
}
Ok(())
}
fn process_mpc_header(
env: &Environment,
header: &Path,
out_file: Option<&Path>,
) -> Result<(), String> {
let mut major = None;
let mut minor = None;
let mut patchlevel = None;
let mut version = None;
let mut reader = open(header);
let mut buf = String::new();
while read_line(&mut reader, &mut buf, header) > 0 {
let s = "#define MPC_VERSION_MAJOR ";
if let Some(start) = buf.find(s) {
major = buf[(start + s.len())..].trim().parse::<i32>().ok();
}
let s = "#define MPC_VERSION_MINOR ";
if let Some(start) = buf.find(s) {
minor = buf[(start + s.len())..].trim().parse::<i32>().ok();
}
let s = "#define MPC_VERSION_PATCHLEVEL ";
if let Some(start) = buf.find(s) {
patchlevel = buf[(start + s.len())..].trim().parse::<i32>().ok();
}
let s = "#define MPC_VERSION_STRING ";
if let Some(start) = buf.find(s) {
version = Some(
buf[(start + s.len())..]
.trim()
.trim_matches('"')
.to_string(),
);
}
buf.clear();
}
drop(reader);
let major = major.expect("Cannot determine MPC_VERSION_MAJOR");
let minor = minor.expect("Cannot determine MPC_VERSION_MINOR");
let patchlevel = patchlevel.expect("Cannot determine MPC_VERSION_PATCHLEVEL");
if !compatible_version(env, major, minor, patchlevel, MPC_VER) {
return Err(format!(
"This version of gmp-mpfr-sys supports MPC {}.{}.{}, but {}.{}.{} was found",
MPC_VER.0, MPC_VER.1, MPC_VER.2, major, minor, patchlevel
));
}
let version = version.expect("Cannot determine MPC_VERSION_STRING");
let content = format!(
concat!(
"const MPC_VERSION_MAJOR: c_int = {};\n",
"const MPC_VERSION_MINOR: c_int = {};\n",
"const MPC_VERSION_PATCHLEVEL: c_int = {};\n",
"const MPC_VERSION_STRING: *const c_char = b\"{}\\0\".as_ptr().cast();\n"
),
major, minor, patchlevel, version
);
if let Some(out_file) = out_file {
let mut rs = create(out_file);
write_flush(&mut rs, &content, out_file);
}
Ok(())
}
fn build_mpfr(env: &Environment, lib: &Path, header: &Path) {
let build_dir = env.build_dir.join("mpfr-build");
create_dir_or_panic(&build_dir);
println!("$ cd {build_dir:?}");
link_dir(
&env.build_dir.join("gmp-build"),
&build_dir.join("gmp-build"),
);
println!("$ unset CC CFLAGS");
std::env::remove_var("CC");
std::env::remove_var("CFLAGS");
let mut conf = String::from(
"../mpfr-src/configure --enable-thread-safe --disable-decimal-float --disable-float128 \
--disable-shared --with-gmp-build=../gmp-build --with-pic",
);
if let Some(cross_target) = env.cross_target.as_ref() {
conf.push_str(" --host ");
conf.push_str(get_actual_cross_target(cross_target));
}
configure(&build_dir, &OsString::from(conf));
make_and_check(env, &build_dir);
let build_lib = build_dir.join("src").join(".libs").join("libmpfr.a");
copy_file_or_panic(&build_lib, lib);
let src_header = env.build_dir.join("mpfr-src").join("src").join("mpfr.h");
copy_file_or_panic(&src_header, header);
}
fn build_mpc(env: &Environment, lib: &Path, header: &Path) {
let build_dir = env.build_dir.join("mpc-build");
create_dir_or_panic(&build_dir);
println!("$ cd {build_dir:?}");
mv("../mpfr-build/gmp-build", &build_dir);
link_dir(&env.build_dir.join("mpfr-src"), &build_dir.join("mpfr-src"));
link_dir(
&env.build_dir.join("mpfr-build"),
&build_dir.join("mpfr-build"),
);
let mut conf = String::from(
"../mpc-src/configure --disable-shared \
--with-mpfr-include=../mpfr-src/src \
--with-mpfr-lib=../mpfr-build/src/.libs \
--with-gmp-include=../gmp-build \
--with-gmp-lib=../gmp-build/.libs --with-pic",
);
if let Some(cross_target) = env.cross_target.as_ref() {
conf.push_str(" --host ");
conf.push_str(get_actual_cross_target(cross_target));
}
configure(&build_dir, &OsString::from(conf));
make_and_check(env, &build_dir);
let build_lib = build_dir.join("src").join(".libs").join("libmpc.a");
copy_file_or_panic(&build_lib, lib);
let src_header = env.build_dir.join("mpc-src").join("src").join("mpc.h");
copy_file_or_panic(&src_header, header);
}
fn write_link_info(env: &Environment, feature_mpfr: bool, feature_mpc: bool) {
let out_str = env.out_dir.to_str().unwrap_or_else(|| {
panic!(
"Path contains unsupported characters, can only make {}",
env.out_dir.display()
)
});
let lib_str = env.lib_dir.to_str().unwrap_or_else(|| {
panic!(
"Path contains unsupported characters, can only make {}",
env.lib_dir.display()
)
});
let include_str = env.include_dir.to_str().unwrap_or_else(|| {
panic!(
"Path contains unsupported characters, can only make {}",
env.include_dir.display()
)
});
println!("cargo:out_dir={out_str}");
println!("cargo:lib_dir={lib_str}");
println!("cargo:include_dir={include_str}");
println!("cargo:rustc-link-search=native={lib_str}");
let target_env = env::var("CARGO_CFG_TARGET_ENV").unwrap_or_default();
if target_env == "musl" && env.use_system_libs {
println!("cargo:rustc-link-search=/usr/lib");
}
let target_features = env::var("CARGO_CFG_TARGET_FEATURE").unwrap_or_default();
let using_static_musl = target_env == "musl" && target_features.contains("crt-static");
let use_static = using_static_musl || !env.use_system_libs;
let maybe_static = if use_static { "static=" } else { "" };
if feature_mpc {
println!("cargo:rustc-link-lib={maybe_static}mpc");
}
if feature_mpfr {
println!("cargo:rustc-link-lib={maybe_static}mpfr");
}
println!("cargo:rustc-link-lib={maybe_static}gmp");
}
impl Environment {
#[allow(dead_code)]
fn check_feature(&self, name: &str, contents: &str, nightly_features: Option<&str>) {
let try_dir = self.out_dir.join(format!("try_{name}"));
let filename = format!("try_{name}.rs");
create_dir_or_panic(&try_dir);
println!("$ cd {try_dir:?}");
enum Iteration {
Stable,
Unstable,
}
for i in &[Iteration::Stable, Iteration::Unstable] {
let s;
let file_contents = match *i {
Iteration::Stable => contents,
Iteration::Unstable => match nightly_features {
Some(features) => {
s = format!("#![feature({features})]\n{contents}");
&s
}
None => continue,
},
};
create_file_or_panic(&try_dir.join(&filename), file_contents);
let mut cmd = Command::new(&self.rustc);
cmd.current_dir(&try_dir)
.stdout(Stdio::null())
.stderr(Stdio::null())
.args([&*filename, "--emit=dep-info,metadata"]);
println!("$ {cmd:?} >& /dev/null");
let status = cmd
.status()
.unwrap_or_else(|_| panic!("Unable to execute: {cmd:?}"));
if status.success() {
println!("cargo:rustc-cfg={name}");
if let Iteration::Unstable = *i {
println!("cargo:rustc-cfg=nightly_{name}");
}
break;
}
}
remove_dir_or_panic(&try_dir);
}
}
fn cargo_env(name: &str) -> OsString {
env::var_os(name)
.unwrap_or_else(|| panic!("environment variable not found: {name}, please use cargo"))
}
fn there_is_env(name: &str) -> bool {
env::var_os(name).is_some()
}
fn check_for_msvc(env: &Environment) {
if env.target == Target::Msvc {
panic!("Windows MSVC target is not supported (linking would fail)");
}
}
#[allow(dead_code)]
fn rustc_later_eq(major: i32, minor: i32) -> bool {
let rustc = cargo_env("RUSTC");
let output = Command::new(rustc)
.arg("--version")
.output()
.expect("unable to run rustc --version");
let version = String::from_utf8(output.stdout).expect("unrecognized rustc version");
if !version.starts_with("rustc ") {
panic!("unrecognized rustc version: {version}");
}
let remain = &version[6..];
let dot = remain.find('.').expect("unrecognized rustc version");
let ver_major = remain[0..dot]
.parse::<i32>()
.expect("unrecognized rustc version");
match ver_major.cmp(&major) {
Ordering::Less => false,
Ordering::Greater => true,
Ordering::Equal => {
let remain = &remain[dot + 1..];
let dot = remain.find('.').expect("unrecognized rustc version");
let ver_minor = remain[0..dot]
.parse::<i32>()
.expect("unrecognized rustc version");
ver_minor >= minor
}
}
}
fn mingw_pkg_config_libdir_or_panic() {
let mut cmd = Command::new("pkg-config");
cmd.args(["--libs-only-L", "--keep-system-libs", "gmp"]);
let output = execute_stdout(cmd);
if output.len() < 2 || &output[0..2] != b"-L" {
panic!("expected pkg-config output to begin with \"-L\"");
}
let libdir = str::from_utf8(&output[2..]).expect("output from pkg-config not utf-8");
println!("cargo:rustc-link-search=native={libdir}");
}
fn remove_dir(dir: &Path) -> IoResult<()> {
if !dir.exists() {
return Ok(());
}
assert!(dir.is_dir(), "Not a directory: {dir:?}");
println!("$ rm -r {dir:?}");
fs::remove_dir_all(dir)
}
fn remove_dir_or_panic(dir: &Path) {
remove_dir(dir).unwrap_or_else(|_| panic!("Unable to remove directory: {dir:?}"));
}
fn create_dir(dir: &Path) -> IoResult<()> {
println!("$ mkdir -p {dir:?}");
fs::create_dir_all(dir)
}
fn create_dir_or_panic(dir: &Path) {
create_dir(dir).unwrap_or_else(|_| panic!("Unable to create directory: {dir:?}"));
}
fn create_file_or_panic(filename: &Path, contents: &str) {
println!("$ printf '%s' {:?}... > {filename:?}", &contents[0..10]);
let mut file =
File::create(filename).unwrap_or_else(|_| panic!("Unable to create file: {filename:?}"));
file.write_all(contents.as_bytes())
.unwrap_or_else(|_| panic!("Unable to write to file: {filename:?}"));
}
fn copy_file(src: &Path, dst: &Path) -> IoResult<u64> {
println!("$ cp {src:?} {dst:?}");
fs::copy(src, dst)
}
fn copy_file_or_panic(src: &Path, dst: &Path) {
copy_file(src, dst).unwrap_or_else(|_| {
panic!("Unable to copy {src:?} -> {dst:?}");
});
}
fn configure(build_dir: &Path, conf_line: &OsStr) {
let mut conf = Command::new("sh");
conf.current_dir(build_dir).arg("-c").arg(conf_line);
execute(conf);
}
fn make_and_check(env: &Environment, build_dir: &Path) {
let mut make = Command::new("make");
make.current_dir(build_dir).arg("-j").arg(&env.jobs);
execute(make);
if !env.c_no_tests && env.cross_target.is_none() {
let mut make_check = Command::new("make");
make_check
.current_dir(build_dir)
.arg("-j")
.arg(&env.jobs)
.arg("check");
execute(make_check);
}
}
#[cfg(unix)]
fn link_dir(src: &Path, dst: &Path) {
println!("$ ln -s {src:?} {dst:?}");
unix_fs::symlink(src, dst).unwrap_or_else(|_| {
panic!("Unable to symlink {src:?} -> {dst:?}");
});
}
#[cfg(windows)]
fn link_dir(src: &Path, dst: &Path) {
println!("$ ln -s {src:?} {dst:?}");
if windows_fs::symlink_dir(src, dst).is_ok() {
return;
}
println!("symlink_dir: failed to create symbolic link, copying instead");
let mut c = Command::new("cp");
c.arg("-R").arg(src).arg(dst);
execute(c);
}
fn mv(src: &str, dst_dir: &Path) {
let mut c = Command::new("mv");
c.arg(src).arg(".").current_dir(dst_dir);
execute(c);
}
fn execute(mut command: Command) {
println!("$ {command:?}");
let status = command
.status()
.unwrap_or_else(|_| panic!("Unable to execute: {command:?}"));
if !status.success() {
if let Some(code) = status.code() {
panic!("Program failed with code {code}: {command:?}");
} else {
panic!("Program failed: {command:?}");
}
}
}
fn execute_stdout(mut command: Command) -> Vec<u8> {
println!("$ {command:?}");
let output = command
.output()
.unwrap_or_else(|_| panic!("Unable to execute: {command:?}"));
if !output.status.success() {
if let Some(code) = output.status.code() {
panic!("Program failed with code {code}: {command:?}");
} else {
panic!("Program failed: {command:?}");
}
}
output.stdout
}
fn open(name: &Path) -> BufReader<File> {
let file = File::open(name).unwrap_or_else(|_| panic!("Cannot open file: {name:?}"));
BufReader::new(file)
}
fn create(name: &Path) -> BufWriter<File> {
let file = File::create(name).unwrap_or_else(|_| panic!("Cannot create file: {name:?}"));
BufWriter::new(file)
}
fn read_line(reader: &mut BufReader<File>, buf: &mut String, name: &Path) -> usize {
reader
.read_line(buf)
.unwrap_or_else(|_| panic!("Cannot read from: {name:?}"))
}
fn write_flush(writer: &mut BufWriter<File>, buf: &str, name: &Path) {
writer
.write_all(buf.as_bytes())
.unwrap_or_else(|_| panic!("Cannot write to: {name:?}"));
writer
.flush()
.unwrap_or_else(|_| panic!("Cannot write to: {name:?}"));
}
fn system_cache_dir() -> Option<PathBuf> {
#[cfg(target_os = "windows")]
{
use core::mem::MaybeUninit;
use core::slice;
use std::os::windows::ffi::OsStringExt;
use windows_sys::Win32::Foundation::S_OK;
use windows_sys::Win32::Globalization;
use windows_sys::Win32::System::Com;
use windows_sys::Win32::UI::Shell::{self, FOLDERID_LocalAppData};
unsafe {
let mut path = MaybeUninit::uninit();
if Shell::SHGetKnownFolderPath(&FOLDERID_LocalAppData, 0, 0, path.as_mut_ptr()) != S_OK
{
return None;
}
let path = path.assume_init();
let slice = slice::from_raw_parts(path, Globalization::lstrlenW(path) as usize);
let string = OsString::from_wide(slice);
Com::CoTaskMemFree(path.cast());
Some(string.into())
}
}
#[cfg(any(target_os = "macos", target_os = "ios"))]
{
env::var_os("HOME")
.filter(|x| !x.is_empty())
.map(|x| AsRef::<Path>::as_ref(&x).join("Library").join("Caches"))
}
#[cfg(not(any(target_os = "windows", target_os = "macos", target_os = "ios")))]
{
env::var_os("XDG_CACHE_HOME")
.filter(|x| !x.is_empty())
.map(PathBuf::from)
.or_else(|| {
env::var_os("HOME")
.filter(|x| !x.is_empty())
.map(|x| AsRef::<Path>::as_ref(&x).join(".cache"))
})
}
}
fn compilation_target_allowed(host: &str, target: &str) -> bool {
if host == target {
return true;
}
let (machine_x86_64, machine_i686) = ("x86_64", "i686");
if host.starts_with(machine_x86_64)
&& target.starts_with(machine_i686)
&& host[machine_x86_64.len()..] == target[machine_i686.len()..]
&& !target.contains("-windows-gnu")
{
return true;
}
false
}
const SYSTEM_GMP_C: &str = r##"/* system_gmp.c */
#include <gmp.h>
#include <stdio.h>
#define STRINGIFY(x) #x
#define DEFINE_STR(x) ("#define " #x " " STRINGIFY(x) "\n")
int main(void) {
FILE *f = fopen("system_gmp.out", "w");
#ifdef _LONG_LONG_LIMB
fputs(DEFINE_STR(_LONG_LONG_LIMB), f);
#else
fputs("#undef _LONG_LONG_LIMB\n", f);
#endif
fputs(DEFINE_STR(__GNU_MP_VERSION), f);
fputs(DEFINE_STR(__GNU_MP_VERSION_MINOR), f);
fputs(DEFINE_STR(__GNU_MP_VERSION_PATCHLEVEL), f);
fputs(DEFINE_STR(GMP_LIMB_BITS), f);
fputs(DEFINE_STR(GMP_NAIL_BITS), f);
fputs(DEFINE_STR(__GMP_CC), f);
fputs(DEFINE_STR(__GMP_CFLAGS), f);
fclose(f);
return 0;
}
"##;
const SYSTEM_MPFR_C: &str = r##"/* system_mpfr.c */
#include <mpfr.h>
#include <stdio.h>
#define STRINGIFY(x) #x
#define DEFINE_STR(x) ("#define " #x " " STRINGIFY(x) "\n")
int main(void) {
FILE *f = fopen("system_mpfr.out", "w");
fputs(DEFINE_STR(MPFR_VERSION_MAJOR), f);
fputs(DEFINE_STR(MPFR_VERSION_MINOR), f);
fputs(DEFINE_STR(MPFR_VERSION_PATCHLEVEL), f);
fputs(DEFINE_STR(MPFR_VERSION_STRING), f);
fclose(f);
return 0;
}
"##;
const SYSTEM_MPC_C: &str = r##"/* system_mpc.c */
#include <mpc.h>
#include <stdio.h>
#define STRINGIFY(x) #x
#define DEFINE_STR(x) ("#define " #x " " STRINGIFY(x) "\n")
int main(void) {
FILE *f = fopen("system_mpc.out", "w");
fputs(DEFINE_STR(MPC_VERSION_MAJOR), f);
fputs(DEFINE_STR(MPC_VERSION_MINOR), f);
fputs(DEFINE_STR(MPC_VERSION_PATCHLEVEL), f);
fputs(DEFINE_STR(MPC_VERSION_STRING), f);
fclose(f);
return 0;
}
"##;