#[cfg(feature = "zig")]
use crate::PlatformTag;
use crate::target::{RUST_1_64_0, RUST_1_93_0};
use crate::{BridgeModel, BuildContext, PythonInterpreter, Target};
use anyhow::{Context, Result, anyhow, bail};
use cargo_metadata::CrateType;
use fat_macho::FatWriter;
use fs_err::{self as fs, File};
use normpath::PathExt;
use std::collections::HashMap;
use std::env;
use std::io::{BufReader, Read};
use std::path::{Path, PathBuf};
use std::process::{Command, Stdio};
use std::str;
use tracing::{debug, instrument, trace};
const PYO3_ABI3_NO_PYTHON_VERSION: (u64, u64, u64) = (0, 16, 4);
pub(crate) const LIB_CRATE_TYPES: [CrateType; 4] = [
CrateType::Lib,
CrateType::DyLib,
CrateType::RLib,
CrateType::StaticLib,
];
#[derive(Debug, Clone)]
pub struct CompileTarget {
pub target: cargo_metadata::Target,
pub bridge_model: BridgeModel,
}
#[derive(Debug, Clone)]
pub struct BuildArtifact {
pub path: PathBuf,
pub linked_paths: Vec<String>,
}
pub fn compile(
context: &BuildContext,
python_interpreter: Option<&PythonInterpreter>,
targets: &[CompileTarget],
) -> Result<Vec<HashMap<CrateType, BuildArtifact>>> {
if context.universal2 {
compile_universal2(context, python_interpreter, targets)
} else {
compile_targets(context, python_interpreter, targets)
}
}
fn compile_universal2(
context: &BuildContext,
python_interpreter: Option<&PythonInterpreter>,
targets: &[CompileTarget],
) -> Result<Vec<HashMap<CrateType, BuildArtifact>>> {
let mut aarch64_context = context.clone();
aarch64_context.target = Target::from_resolved_target_triple("aarch64-apple-darwin")?;
let aarch64_artifacts = compile_targets(&aarch64_context, python_interpreter, targets)
.context("Failed to build a aarch64 library through cargo")?;
let mut x86_64_context = context.clone();
x86_64_context.target = Target::from_resolved_target_triple("x86_64-apple-darwin")?;
let x86_64_artifacts = compile_targets(&x86_64_context, python_interpreter, targets)
.context("Failed to build a x86_64 library through cargo")?;
let mut universal_artifacts = Vec::with_capacity(targets.len());
for (bridge_model, (aarch64_artifact, x86_64_artifact)) in targets
.iter()
.map(|target| &target.bridge_model)
.zip(aarch64_artifacts.iter().zip(&x86_64_artifacts))
{
let build_type = if bridge_model.is_bin() {
CrateType::Bin
} else {
CrateType::CDyLib
};
let aarch64_artifact = aarch64_artifact.get(&build_type).cloned().ok_or_else(|| {
if build_type == CrateType::CDyLib {
anyhow!(
"Cargo didn't build an aarch64 cdylib. Did you miss crate-type = [\"cdylib\"] \
in the lib section of your Cargo.toml?",
)
} else {
anyhow!("Cargo didn't build an aarch64 bin.")
}
})?;
let x86_64_artifact = x86_64_artifact.get(&build_type).cloned().ok_or_else(|| {
if build_type == CrateType::CDyLib {
anyhow!(
"Cargo didn't build a x86_64 cdylib. Did you miss crate-type = [\"cdylib\"] \
in the lib section of your Cargo.toml?",
)
} else {
anyhow!("Cargo didn't build a x86_64 bin.")
}
})?;
let output_path = aarch64_artifact
.path
.display()
.to_string()
.replace("aarch64-apple-darwin/", "");
let mut writer = FatWriter::new();
let aarch64_file = fs::read(&aarch64_artifact.path)?;
let x86_64_file = fs::read(&x86_64_artifact.path)?;
writer
.add(aarch64_file)
.map_err(|e| anyhow!("Failed to add aarch64 cdylib: {:?}", e))?;
writer
.add(x86_64_file)
.map_err(|e| anyhow!("Failed to add x86_64 cdylib: {:?}", e))?;
writer
.write_to_file(&output_path)
.map_err(|e| anyhow!("Failed to create universal cdylib: {:?}", e))?;
let mut result = HashMap::new();
let universal_artifact = BuildArtifact {
path: PathBuf::from(output_path),
..x86_64_artifact
};
result.insert(build_type, universal_artifact);
universal_artifacts.push(result);
}
Ok(universal_artifacts)
}
fn compile_targets(
context: &BuildContext,
python_interpreter: Option<&PythonInterpreter>,
targets: &[CompileTarget],
) -> Result<Vec<HashMap<CrateType, BuildArtifact>>> {
let mut artifacts = Vec::with_capacity(targets.len());
for target in targets {
let build_command = cargo_build_command(context, python_interpreter, target)?;
artifacts.push(compile_target(context, build_command)?);
}
Ok(artifacts)
}
fn cargo_build_command(
context: &BuildContext,
python_interpreter: Option<&PythonInterpreter>,
compile_target: &CompileTarget,
) -> Result<Command> {
let target = &context.target;
let user_specified_target = if target.user_specified {
Some(target.target_triple().to_string())
} else {
None
};
let mut cargo_rustc = context
.cargo_options
.clone()
.into_rustc_options(user_specified_target);
cargo_rustc.message_format = vec!["json-render-diagnostics".to_string()];
if compile_target
.target
.crate_types
.iter()
.any(|crate_type| LIB_CRATE_TYPES.contains(crate_type))
{
if target.rustc_version.semver >= RUST_1_64_0 {
debug!("Setting crate_type to cdylib for Rust >= 1.64.0");
cargo_rustc.crate_type = vec!["cdylib".to_string()];
}
}
let target_triple = target.target_triple();
let manifest_dir = context.manifest_path.parent().unwrap();
let mut rustflags = cargo_config2::Config::load_with_cwd(manifest_dir)?
.rustflags(target_triple)?
.unwrap_or_default();
let original_rustflags = rustflags.flags.clone();
let bridge_model = &compile_target.bridge_model;
match bridge_model {
BridgeModel::Bin(..) => {
cargo_rustc.bin.push(compile_target.target.name.clone());
}
BridgeModel::Cffi | BridgeModel::UniFfi | BridgeModel::PyO3 { .. } => {
cargo_rustc.lib = true;
if context.target.is_musl_libc()
&& !rustflags
.flags
.iter()
.any(|f| f == "target-feature=-crt-static")
{
debug!("Setting `-C target-features=-crt-static` for musl dylib");
rustflags.push("-C");
rustflags.push("target-feature=-crt-static");
}
}
}
if target.is_macos() {
if let BridgeModel::PyO3 { .. } = bridge_model {
let module_name = &context.module_name;
let so_filename = if bridge_model.is_abi3() {
format!("{module_name}.abi3.so")
} else {
python_interpreter
.expect("missing python interpreter for non-abi3 wheel build")
.get_library_name(module_name)
};
let macos_dylib_install_name =
format!("link-args=-Wl,-install_name,@rpath/{so_filename}");
let mac_args = [
"-C".to_string(),
"link-arg=-undefined".to_string(),
"-C".to_string(),
"link-arg=dynamic_lookup".to_string(),
"-C".to_string(),
macos_dylib_install_name,
];
debug!("Setting additional linker args for macOS: {:?}", mac_args);
cargo_rustc.args.extend(mac_args);
}
} else if target.is_emscripten() {
if target.rustc_version.semver < RUST_1_93_0
&& !rustflags
.flags
.iter()
.any(|f| f.contains("link-native-libraries"))
{
debug!("Setting `-Z link-native-libraries=no` for Emscripten (rust < 1.93.0)");
rustflags.push("-Z");
rustflags.push("link-native-libraries=no");
}
let mut emscripten_args = Vec::new();
if !cargo_rustc
.args
.iter()
.any(|arg| arg.contains("SIDE_MODULE"))
{
emscripten_args.push("-C".to_string());
emscripten_args.push("link-arg=-sSIDE_MODULE=2".to_string());
}
if !cargo_rustc
.args
.iter()
.any(|arg| arg.contains("WASM_BIGINT"))
{
emscripten_args.push("-C".to_string());
emscripten_args.push("link-arg=-sWASM_BIGINT".to_string());
}
debug!(
"Setting additional linker args for Emscripten: {:?}",
emscripten_args
);
cargo_rustc.args.extend(emscripten_args);
}
if context.strip {
cargo_rustc
.args
.extend(["-C".to_string(), "strip=symbols".to_string()]);
}
let mut build_command = if target.is_msvc() && target.cross_compiling() {
#[cfg(feature = "xwin")]
{
let native_compile = target.host_triple().contains("windows-msvc")
&& cc::Build::new()
.opt_level(0)
.host(target.host_triple())
.target(target_triple)
.cargo_metadata(false)
.cargo_warnings(false)
.cargo_output(false)
.try_get_compiler()
.is_ok();
let force_xwin = env::var("MATURIN_USE_XWIN").ok().as_deref() == Some("1");
if !native_compile || force_xwin {
println!("🛠️ Using xwin for cross-compiling to {target_triple}");
let xwin_options = {
use clap::Parser;
cargo_xwin::XWinOptions::parse_from(Vec::<&str>::new())
};
let mut build = cargo_xwin::Rustc::from(cargo_rustc);
build.target = vec![target_triple.to_string()];
build.xwin = xwin_options;
build.build_command()?
} else {
if target.user_specified {
cargo_rustc.target = vec![target_triple.to_string()];
}
cargo_rustc.command()
}
}
#[cfg(not(feature = "xwin"))]
{
if target.user_specified {
cargo_rustc.target = vec![target_triple.to_string()];
}
cargo_rustc.command()
}
} else {
#[cfg(feature = "zig")]
{
let mut build = cargo_zigbuild::Rustc::from(cargo_rustc);
if !context.zig {
build.disable_zig_linker = true;
if target.user_specified {
build.target = vec![target_triple.to_string()];
}
} else {
println!("🛠️ Using zig for cross-compiling to {target_triple}");
build.enable_zig_ar = true;
let zig_triple = if target.is_linux() && !target.is_musl_libc() {
match context.platform_tag.iter().find(|tag| tag.is_manylinux()) {
Some(PlatformTag::Manylinux { major, minor }) => {
format!("{target_triple}.{major}.{minor}")
}
_ => target_triple.to_string(),
}
} else {
target_triple.to_string()
};
build.target = vec![zig_triple];
}
build.build_command()?
}
#[cfg(not(feature = "zig"))]
{
if target.user_specified {
cargo_rustc.target = vec![target_triple.to_string()];
}
cargo_rustc.command()
}
};
#[cfg(feature = "zig")]
if context.zig {
if let Ok((zig_cmd, zig_args)) = cargo_zigbuild::Zig::find_zig() {
if zig_args.is_empty() {
build_command.env("ZIG_COMMAND", zig_cmd);
} else {
build_command.env(
"ZIG_COMMAND",
format!("{} {}", zig_cmd.display(), zig_args.join(" ")),
);
};
}
}
build_command
.stdout(Stdio::piped())
.stderr(Stdio::inherit());
if !rustflags.flags.is_empty() && rustflags.flags != original_rustflags {
build_command.env("CARGO_ENCODED_RUSTFLAGS", rustflags.encode()?);
}
if bridge_model.is_abi3() {
let is_pypy_or_graalpy = python_interpreter
.map(|p| p.interpreter_kind.is_pypy() || p.interpreter_kind.is_graalpy())
.unwrap_or(false);
if !is_pypy_or_graalpy && !target.is_windows() {
let pyo3_ver = pyo3_version(&context.cargo_metadata)
.context("Failed to get pyo3 version from cargo metadata")?;
if pyo3_ver < PYO3_ABI3_NO_PYTHON_VERSION {
build_command.env("PYO3_NO_PYTHON", "1");
}
}
}
if bridge_model.is_pyo3() && !bridge_model.is_bin() {
build_command.env("PYO3_BUILD_EXTENSION_MODULE", "1");
}
if let Some(interpreter) = python_interpreter {
if interpreter.runnable {
if bridge_model.is_pyo3() {
debug!(
"Setting PYO3_PYTHON to {}",
interpreter.executable.display()
);
build_command
.env("PYO3_PYTHON", &interpreter.executable)
.env(
"PYO3_ENVIRONMENT_SIGNATURE",
interpreter.environment_signature(),
);
}
build_command.env("PYTHON_SYS_EXECUTABLE", &interpreter.executable);
} else if bridge_model.is_pyo3() && env::var_os("PYO3_CONFIG_FILE").is_none() {
let pyo3_config = interpreter.pyo3_config_file();
let maturin_target_dir = context.target_dir.join(env!("CARGO_PKG_NAME"));
let config_file = maturin_target_dir.join(format!(
"pyo3-config-{}-{}.{}.txt",
target_triple, interpreter.major, interpreter.minor
));
fs::create_dir_all(&maturin_target_dir)?;
let existing_pyo3_config = fs::read_to_string(&config_file).unwrap_or_default();
if pyo3_config != existing_pyo3_config {
fs::write(&config_file, pyo3_config).with_context(|| {
format!(
"Failed to create pyo3 config file at '{}'",
config_file.display()
)
})?;
}
let abs_config_file = config_file.normalize()?.into_path_buf();
build_command.env("PYO3_CONFIG_FILE", abs_config_file);
}
}
if !context.editable && target.is_macos() && env::var_os("MACOSX_DEPLOYMENT_TARGET").is_none() {
use crate::build_context::rustc_macosx_target_version;
let target_config = context
.pyproject_toml
.as_ref()
.and_then(|x| x.target_config(target_triple));
let deployment_target = if let Some(deployment_target) = target_config
.as_ref()
.and_then(|config| config.macos_deployment_target.as_ref())
{
eprintln!(
"💻 Using `MACOSX_DEPLOYMENT_TARGET={deployment_target}` for {target_triple} by configuration"
);
deployment_target.clone()
} else {
let (major, minor) = rustc_macosx_target_version(target_triple);
eprintln!(
"💻 Using `MACOSX_DEPLOYMENT_TARGET={major}.{minor}` for {target_triple} by default"
);
format!("{major}.{minor}")
};
build_command.env("MACOSX_DEPLOYMENT_TARGET", deployment_target);
}
Ok(build_command)
}
fn compile_target(
context: &BuildContext,
mut build_command: Command,
) -> Result<HashMap<CrateType, BuildArtifact>> {
debug!("Running {:?}", build_command);
let using_cross = build_command
.get_program()
.to_string_lossy()
.starts_with("cross");
let mut cargo_build = build_command
.spawn()
.context("Failed to run `cargo rustc`")?;
let mut artifacts = HashMap::new();
let mut linked_paths = Vec::new();
let stream = cargo_build
.stdout
.take()
.expect("Cargo build should have a stdout");
for message in cargo_metadata::Message::parse_stream(BufReader::new(stream)) {
let message = message.context("Failed to parse cargo metadata message")?;
trace!("cargo message: {:?}", message);
match message {
cargo_metadata::Message::CompilerArtifact(artifact) => {
let package_in_metadata = context
.cargo_metadata
.packages
.iter()
.find(|package| package.id == artifact.package_id);
let crate_name = match package_in_metadata {
Some(package) => &package.name,
None => {
let package_id = &artifact.package_id;
let should_warn = !package_id.repr.contains("rustup")
&& !package_id.repr.contains("rustlib")
&& !artifact.features.contains(&"rustc-dep-of-std".to_string());
if should_warn {
eprintln!(
"⚠️ Warning: The package {package_id} wasn't listed in `cargo metadata`"
);
}
continue;
}
};
if crate_name.as_ref() == context.crate_name {
let tuples = artifact
.target
.crate_types
.into_iter()
.zip(artifact.filenames);
for (crate_type, filename) in tuples {
let path = if using_cross && filename.starts_with("/target") {
context
.cargo_metadata
.target_directory
.join(filename.strip_prefix("/target").unwrap())
.into_std_path_buf()
} else {
filename.into()
};
let artifact = BuildArtifact {
path,
linked_paths: Vec::new(),
};
artifacts.insert(crate_type, artifact);
}
}
}
cargo_metadata::Message::BuildScriptExecuted(msg) => {
let has_dylib = msg.linked_libs.iter().map(|l| l.as_str()).any(|lib| {
if let Some(index) = lib.find('=') {
let kind = &lib[..index];
kind.starts_with("dylib")
} else {
true
}
});
if has_dylib {
for path in msg.linked_paths.iter().map(|p| p.as_str()) {
if let Some(index) = path.find('=') {
let kind = &path[..index];
if kind == "native" || kind == "all" {
linked_paths.push(path[index + 1..].to_string());
}
} else {
linked_paths.push(path.to_string());
}
}
}
}
cargo_metadata::Message::CompilerMessage(msg) => {
println!("{}", msg.message);
}
_ => (),
}
}
for artifact in artifacts.values_mut() {
artifact.linked_paths.clone_from(&linked_paths);
}
let status = cargo_build
.wait()
.expect("Failed to wait on cargo child process");
if !status.success() {
bail!(
r#"Cargo build finished with "{}": `{:?}`"#,
status,
build_command,
)
}
Ok(artifacts)
}
#[instrument(skip_all)]
pub fn warn_missing_py_init(artifact: &Path, module_name: &str) -> Result<()> {
let py_init = format!("PyInit_{module_name}");
let mut fd = File::open(artifact)?;
let mut buffer = Vec::new();
fd.read_to_end(&mut buffer)?;
let mut found = false;
match goblin::Object::parse(&buffer)? {
goblin::Object::Elf(elf) => {
for dyn_sym in elf.dynsyms.iter() {
if py_init == elf.dynstrtab[dyn_sym.st_name] {
found = true;
break;
}
}
}
goblin::Object::Mach(mach) => {
match mach {
goblin::mach::Mach::Binary(macho) => {
for sym in macho.exports()? {
let sym_name = sym.name;
if py_init == sym_name.strip_prefix('_').unwrap_or(&sym_name) {
found = true;
break;
}
}
if !found {
for sym in macho.symbols() {
let (sym_name, _) = sym?;
if py_init == sym_name.strip_prefix('_').unwrap_or(sym_name) {
found = true;
break;
}
}
}
}
goblin::mach::Mach::Fat(_) => {
found = true
}
}
}
goblin::Object::PE(pe) => {
for sym in &pe.exports {
if let Some(sym_name) = sym.name
&& py_init == sym_name
{
found = true;
break;
}
}
}
_ => {
found = true
}
}
if !found {
eprintln!(
"⚠️ Warning: Couldn't find the symbol `{py_init}` in the native library. \
Python will fail to import this module. \
If you're using pyo3, check that `#[pymodule]` uses `{module_name}` as module name"
)
}
Ok(())
}
fn pyo3_version(cargo_metadata: &cargo_metadata::Metadata) -> Option<(u64, u64, u64)> {
let packages: HashMap<&str, &cargo_metadata::Package> = cargo_metadata
.packages
.iter()
.filter_map(|pkg| {
let name = pkg.name.as_ref();
if name == "pyo3" || name == "pyo3-ffi" {
Some((name, pkg))
} else {
None
}
})
.collect();
packages
.get("pyo3")
.or_else(|| packages.get("pyo3-ffi"))
.map(|pkg| (pkg.version.major, pkg.version.minor, pkg.version.patch))
}