Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Refactor cross compile assistance #769

Merged
merged 28 commits into from
Feb 8, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
2da914b
Refactor cross compilation assistance
runesoerensen Feb 1, 2024
47ddc1f
No need to assign linker value to variable
runesoerensen Feb 1, 2024
99ac631
Prefer pattern matching
runesoerensen Feb 1, 2024
276b1e7
Simplify expression
runesoerensen Feb 1, 2024
52c6ec1
Move error handling logic
runesoerensen Feb 1, 2024
533665e
Combine functions
runesoerensen Feb 1, 2024
5954ba8
Provide better help text
runesoerensen Feb 1, 2024
84b4d8c
Update help text to better fit output
runesoerensen Feb 1, 2024
dcb5f12
Decrease help text verbosity level
runesoerensen Feb 1, 2024
de1aec0
Use underscores in CC_* env vars
runesoerensen Feb 1, 2024
a7510c8
Fully qualify OS and ARCH name references
runesoerensen Feb 2, 2024
cc5af13
Avoid declaring new target variable
runesoerensen Feb 2, 2024
cab120b
Explicitly match supported mac archs
runesoerensen Feb 2, 2024
81d9bf3
Add indoc dependency to libcnb-package
runesoerensen Feb 2, 2024
6d02966
Reduce line length
runesoerensen Feb 2, 2024
0504c41
Use indoc for multiline help text readability
runesoerensen Feb 2, 2024
1d3670b
Prefer formatdoc macro
runesoerensen Feb 2, 2024
d2b7be7
Add whitespace
runesoerensen Feb 2, 2024
aa2ea7c
Add comment explaining musl-gcc env logic
runesoerensen Feb 2, 2024
7b170d3
Don't use raw string literal
runesoerensen Feb 2, 2024
32cce40
Remove newline
runesoerensen Feb 2, 2024
8e53c80
Fix gcc-x86-64 compiler package name
runesoerensen Feb 2, 2024
c5a046b
Add required packages to help text
runesoerensen Feb 2, 2024
bffcc40
Add required amd64 packages
runesoerensen Feb 2, 2024
c32607a
Update changelog `libcnb-package` changes
runesoerensen Feb 5, 2024
e9bcd4c
Adjust help text wrapping
runesoerensen Feb 8, 2024
e040fee
Rename gcc_path to gcc_binary_name
runesoerensen Feb 8, 2024
f9fb1cf
Move changelog entries to top level
runesoerensen Feb 8, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 2 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -11,6 +11,8 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0

### Changed

- Improved the consistency of cross-compilation assistance provided across all supported `target_triple` and host OS/architecture combinations. [#769](https://github.com/heroku/libcnb.rs/pull/769)
- Added cross-compilation assistance for `aarch64-unknown-linux-musl` (on macOS and ARM64 Linux) and `x86_64-unknown-linux-musl` (on ARM64 Linux). [#769](https://github.com/heroku/libcnb.rs/pull/769)
- `libcnb`:
- Changed `Layer` interface from `&self` to `&mut self`. ([#669](https://github.com/heroku/libcnb.rs/pull/669))

Expand Down
1 change: 1 addition & 0 deletions libcnb-package/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ workspace = true
[dependencies]
cargo_metadata = "0.18.1"
ignore = "0.4"
indoc = "2.0.4"
libcnb-common.workspace = true
libcnb-data.workspace = true
petgraph = { version = "0.6.4", default-features = false }
Expand Down
151 changes: 77 additions & 74 deletions libcnb-package/src/cross_compile.rs
Original file line number Diff line number Diff line change
@@ -1,3 +1,5 @@
use indoc::{formatdoc, indoc};
use std::env::consts;
use std::ffi::OsString;
use which::which;

Expand All @@ -7,95 +9,89 @@ use which::which;
/// look for the required tools and returns a human-readable help text if they can't be found or
/// any other issue has been detected.
pub fn cross_compile_assistance(target_triple: impl AsRef<str>) -> CrossCompileAssistance {
// Background: https://omarkhawaja.com/cross-compiling-rust-from-macos-to-linux/
if target_triple.as_ref() == X86_64_UNKNOWN_LINUX_MUSL && cfg!(target_os = "macos") {
// There is more than just one binary name here since we also support the binary name for
// an older version cross_compile_assistance which suggested installing a different
// toolchain.
let possible_gcc_binary_names = ["x86_64-unknown-linux-musl-gcc", "x86_64-linux-musl-gcc"];
let target_triple = target_triple.as_ref();
let (gcc_binary_name, help_text) = match (target_triple, consts::OS, consts::ARCH) {
(AARCH64_UNKNOWN_LINUX_MUSL, OS_LINUX, ARCH_X86_64) => (
"aarch64-linux-gnu-gcc",
indoc! {"
To install an aarch64 cross-compiler on Ubuntu:
sudo apt-get install g++-aarch64-linux-gnu libc6-dev-arm64-cross musl-tools
"},
),
(AARCH64_UNKNOWN_LINUX_MUSL, OS_MACOS, ARCH_X86_64 | ARCH_AARCH64) => (
"aarch64-unknown-linux-musl-gcc",
indoc! {"
To install an aarch64 cross-compiler on macOS:
brew install messense/macos-cross-toolchains/aarch64-unknown-linux-musl
"},
),
(AARCH64_UNKNOWN_LINUX_MUSL, OS_LINUX, ARCH_AARCH64)
| (X86_64_UNKNOWN_LINUX_MUSL, OS_LINUX, ARCH_X86_64) => (
"musl-gcc",
indoc! {"
To install musl-tools on Ubuntu:
sudo apt-get install musl-tools
"},
),
(X86_64_UNKNOWN_LINUX_MUSL, OS_LINUX, ARCH_AARCH64) => (
"x86_64-linux-gnu-gcc",
indoc! {"
To install an x86_64 cross-compiler on Ubuntu:
sudo apt-get install g++-x86-64-linux-gnu libc6-dev-amd64-cross musl-tools
"},
),
(X86_64_UNKNOWN_LINUX_MUSL, OS_MACOS, ARCH_X86_64 | ARCH_AARCH64) => (
"x86_64-unknown-linux-musl-gcc",
indoc! {"
To install an x86_64 cross-compiler on macOS:
brew install messense/macos-cross-toolchains/x86_64-unknown-linux-musl
"},
),
_ => return CrossCompileAssistance::NoAssistance,
};

possible_gcc_binary_names
.iter()
.find_map(|binary_name| which(binary_name).ok())
.map_or_else(|| CrossCompileAssistance::HelpText(String::from(
r"For cross-compilation from macOS to x86_64-unknown-linux-musl, a C compiler and
linker for the target platform must be installed on your computer.

The easiest way to install the required cross-compilation toolchain is to run:
brew install messense/macos-cross-toolchains/x86_64-unknown-linux-musl

For more information, see:
https://github.com/messense/homebrew-macos-cross-toolchains

You will also need to install the Rust target, using:
rustup target add x86_64-unknown-linux-musl",
)), |gcc_binary_path| {
match which(gcc_binary_name) {
Ok(_) => {
// When the gcc binary name is `musl-gcc`, Cargo will automatically select the appropriate default linker,
// and set the required environment variables.
if gcc_binary_name == "musl-gcc" {
CrossCompileAssistance::Configuration {
cargo_env: Vec::new(),
}
} else {
CrossCompileAssistance::Configuration {
cargo_env: vec![
(
// Required until Cargo can auto-detect the musl-cross gcc/linker itself,
// since otherwise it checks for a binary named 'musl-gcc':
// https://github.com/FiloSottile/homebrew-musl-cross/issues/16
// since otherwise it checks for a binary named 'musl-gcc' (which is handled above):
// https://github.com/rust-lang/cargo/issues/4133
OsString::from("CARGO_TARGET_X86_64_UNKNOWN_LINUX_MUSL_LINKER"),
OsString::from(&gcc_binary_path),
OsString::from(format!(
"CARGO_TARGET_{}_LINKER",
target_triple.to_uppercase().replace('-', "_")
)),
OsString::from(gcc_binary_name),
),
(
// Required so that any crates that call out to gcc are also cross-compiled:
// https://github.com/alexcrichton/cc-rs/issues/82
OsString::from("CC_x86_64_unknown_linux_musl"),
OsString::from(&gcc_binary_path),
OsString::from(format!("CC_{}", target_triple.replace('-', "_"))),
OsString::from(gcc_binary_name),
),
],
}
})
} else if target_triple.as_ref() == X86_64_UNKNOWN_LINUX_MUSL && cfg!(target_os = "linux") {
match which("musl-gcc") {
Ok(_) => CrossCompileAssistance::Configuration {
cargo_env: Vec::new(),
},
Err(_) => CrossCompileAssistance::HelpText(String::from(
r"For cross-compilation from Linux to x86_64-unknown-linux-musl, a C compiler and
linker for the target platform must be installed on your computer.

The easiest way to install 'musl-gcc' is to install the 'musl-tools' package:
- https://packages.ubuntu.com/focal/musl-tools
- https://packages.debian.org/bullseye/musl-tools

You will also need to install the Rust target, using:
rustup target add x86_64-unknown-linux-musl",
)),
}
}
} else if target_triple.as_ref() == AARCH64_UNKNOWN_LINUX_MUSL && cfg!(target_os = "linux") {
let gcc_binary_path = "aarch64-linux-gnu-gcc";
match which(gcc_binary_path) {
Ok(_) => CrossCompileAssistance::Configuration {
cargo_env: vec![
(
OsString::from("CARGO_TARGET_AARCH64_UNKNOWN_LINUX_MUSL_LINKER"),
OsString::from(gcc_binary_path),
),
(
OsString::from("CC_aarch64_unknown_linux_musl"),
OsString::from(gcc_binary_path),
),
],
},
Err(_) => CrossCompileAssistance::HelpText(String::from(
r"For cross-compilation from Linux to aarch64-unknown-linux-musl, a C compiler and
linker for the target platform must installed on your computer.

The easiest way to install the 'g++-aarch64-linux-gnu', 'libc6-dev-arm64-cross', and 'musl-tools' packages:
- https://packages.ubuntu.com/focal/g++-aarch64-linux-gnu
- https://packages.ubuntu.com/focal/musl-tools
- https://packages.ubuntu.com/focal/libc6-dev-arm64-cross
Err(_) => CrossCompileAssistance::HelpText(formatdoc! {"
For cross-compilation from {0} {1} to {target_triple},
a C compiler and linker for the target platform must be installed:

You will also need to install the Rust target, using:
rustup target add aarch64-unknown-linux-musl",
)),
}
} else {
CrossCompileAssistance::NoAssistance
{help_text}
You will also need to install the Rust target:
rustup target add {target_triple}
",
consts::ARCH,
consts::OS
}),
}
}

Expand All @@ -111,5 +107,12 @@ pub enum CrossCompileAssistance {
},
}

// Constants for supported target triples
const AARCH64_UNKNOWN_LINUX_MUSL: &str = "aarch64-unknown-linux-musl";
const X86_64_UNKNOWN_LINUX_MUSL: &str = "x86_64-unknown-linux-musl";

// Constants for `std::env::consts::OS` and `std::env::consts::ARCH`
const OS_LINUX: &str = "linux";
const OS_MACOS: &str = "macos";
const ARCH_X86_64: &str = "x86_64";
const ARCH_AARCH64: &str = "aarch64";