-
Notifications
You must be signed in to change notification settings - Fork 84
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
Adding CrateId Metadata Hash method for Bazel #365
Conversation
@irajtaghlidi can you add a description to the PR, describing what changes are being made (at a high level) and why they are being made. |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@irajtaghlidi
Thank you for creating this PR.
As @Pagten mentioned, could you add summary of changes in this PR in PR description?
Also, I want to learn more about the context? I assume these changes are used for build this crate with Bazel. Could you provide some links to document about it.
ContextHere's some additional context information about this PR. rustc takes a -Cmetadata argument. From this argument and other properties, rustc calculates a "Stable Crate Id" hash, which it uses for a variety of things, including symbol name mangling. Cargo generates the -Cmetadata argument for rustc based on a variety of factors, like the package version, source, and the target kind. Rust-mbedtls defines a couple of symbols in C code. To allow a crate to depend on multiple versions of rust-mbedtls, all symbols need to be unique. Rust-mbedtls ensures this uniqueness for the C symbols by (at build time, using its build script) fetching the -Cmetadata value generated by Cargo and appending that value to the C code symbol names. Unfortunately, there is no "official" way for a build script to get the -Cmetadata value from Cargo. To work around this, the rust-mbedtls build script extracts the medata value from the crate's directory path in the target directory, via the However, because the structure of the
This is one of the reasons rust-mbedtls currently cannot be compiled using Bazel and that is what this PR tries to address. Approach of the PRThe current approach of the PR is to have the rust-mbedtls build script fall back to a different way of obtaining a -Cmetadata value when it detects that it is being compiled using Bazel. The way that the value is being calculated is:
Review commentsI think the use of The metadata value that is generated in the current version of the PR is based only on the compiler version and the crate version. The value generated by Cargo is based on a lot more components: the crate name, the source of the crate (crates.io or otherwise), enabled feature flags, metadata of the crate's dependencies, compilation profile, compilation target, etc. If the goal is to generate a generic metadata value that can be used for various purposes, all of these components need to be included into it. But the rust-mbedtls build script only uses the metadata value for a specific purpose: to make the C code symbols unique, to allow multiple versions of the crate to be included in the same binary. Hence it can be okay to base the symbol suffix on fewer components. Cargo's dependency resolution algorithm unifies dependencies that have the same crate name, a semver-compatible version and the same source (the source is, e.g., crates.io, git, local, ...). So, for Cargo builds, it would be sufficient to base the symbol suffix on the crate name, crate version and the crate's source. Rules_rust's dependency resolution algorithm is slightly different from that of Cargo. It does not take the source of a crate into account for dependency unification. This means it is not possible to have, for example, a version of a crate from crates.io in the dependency graph together with the same version of the same crate from github (this may be considered a bug/limitation of rules_rust). So, for Bazel builds, it is sufficient to base the symbol suffix on the crate name and crate version. Therefore, I propose we make the following changes:
Happy to hear your thought on this, @irajtaghlidi and other reviewers. |
I agree with Pieter on the shortcomings mentioned. Can you add a comment in |
Indeed, depending on how mbedtls is configured, That said, does rust-mbedtls really support mbedtls configurations where
|
bors r+ |
bors r+ |
@Taowyoo I see you've merge all recent PRs on this repo. Aren't we using bors anymore? |
Ah I see the bors app is no longer running: https://github.com/apps/bors . @Taowyoo can you merge this PR to master? |
That is the intention, yes. |
mbedtls/build.rs
Outdated
let crate_ = out_dir_it.next().unwrap().to_string_lossy(); | ||
assert!(crate_.starts_with("mbedtls-")); | ||
return crate_[8..].to_owned(); | ||
} else if next == "_bs.out_dir" { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Is this check correct?
When I do a build using Bazel, my OUT_DIR env var is set as, e.g.,:
OUT_DIR='${pwd}/bazel-out/k8-fastbuild/bin/external/crate_index__mbedtls-sys-auto-2.26.1/mbedtls-sys-auto_bs.out_dir'
There is no _bs.out_dir
component in that path.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I've never seen that format. In my setup with the latest version of bazel/Rules_rust the OUT_DIR is in this format:
OUT_DIR='${pwd}/bazel-out/k8-fastbuild/bin/external/crate_index__mbedtls-0.12.3/_bs.out_dir' \
Could you please provide your WORKSPACE/Branch to reproduce it?
Hi @Pagten , I guess there are still some comments need to be resolved? |
Yes, |
mbedtls/build.rs
Outdated
let path_to_hash = Path::new(out_dir_str.as_ref()); | ||
let parts_to_hash = path_to_hash | ||
.to_str() |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This seems unnecessary.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Partially left from the previous method. Fixed now.
mbedtls/build.rs
Outdated
let last_part = out_dir_it.next().unwrap(); | ||
|
||
if last_part == "out" { | ||
// If Cargo is used as build system. | ||
let crate_ = out_dir_it.next().unwrap().to_string_lossy(); | ||
assert!(crate_.starts_with("mbedtls-"), "Expected directory to start with 'mbedtls-'"); | ||
return crate_[8..].to_owned(); // Return the part after "mbedtls-" | ||
} else if out_dir_str.contains("bazel-out") { | ||
// If Bazel is used as build system. | ||
let parts_to_hash = out_dir_str | ||
.split("bazel-out") | ||
.nth(1) | ||
.expect("Invalid Bazel out dir structure"); | ||
|
||
let string_to_hash = format!("bazel-out{}", parts_to_hash); | ||
let mut hasher = DefaultHasher::new(); | ||
string_to_hash.hash(&mut hasher); | ||
let hash = hasher.finish(); | ||
return format!("{:x}", hash); | ||
} else { | ||
panic!("Unexpected directory structure: {:?}", out_dir_str); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
let last_part = out_dir_it.next().unwrap(); | |
if last_part == "out" { | |
// If Cargo is used as build system. | |
let crate_ = out_dir_it.next().unwrap().to_string_lossy(); | |
assert!(crate_.starts_with("mbedtls-"), "Expected directory to start with 'mbedtls-'"); | |
return crate_[8..].to_owned(); // Return the part after "mbedtls-" | |
} else if out_dir_str.contains("bazel-out") { | |
// If Bazel is used as build system. | |
let parts_to_hash = out_dir_str | |
.split("bazel-out") | |
.nth(1) | |
.expect("Invalid Bazel out dir structure"); | |
let string_to_hash = format!("bazel-out{}", parts_to_hash); | |
let mut hasher = DefaultHasher::new(); | |
string_to_hash.hash(&mut hasher); | |
let hash = hasher.finish(); | |
return format!("{:x}", hash); | |
} else { | |
panic!("Unexpected directory structure: {:?}", out_dir_str); | |
let mut out_dir_it_rev = out_dir.iter().rev(); | |
let mut out_dir_it = out_dir.iter(); | |
if out_dir_it_rev.next().map_or(false, |p| p == "out") { | |
// If Cargo is used as build system. | |
let crate_ = out_dir_it_rev.next().unwrap().to_string_lossy(); | |
assert!(crate_.starts_with("mbedtls-"), "Expected directory to start with 'mbedtls-'"); | |
return crate_[8..].to_owned(); // Return the part after "mbedtls-" | |
} else if out_dir_it.position(|p| p == "bazel-out").is_some() { | |
// If Bazel is used as build system. | |
let mut hasher = DefaultHasher::new(); | |
for p in out_dir_it { | |
p.hash(&mut hasher); | |
} | |
let hash = hasher.finish(); | |
return format!("{:016x}", hash); | |
} else { | |
panic!("unexpected OUT_DIR format: {}", out_dir.to_string_lossy()); |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Much better implementation. Thanks.
# Configuration: c2b2998e956e739e6041daea0eaadcf22e7f68c3e819bfbfc43c92b444190896
# Execution platform: //:docker_image_platform
[237 / 238] Compiling Rust bin service1 (1 files); 0s remote, remote-cache
[237 / 238] Compiling Rust bin service1 (1 files); 10s remote, remote-cache
[237 / 238] Compiling Rust bin service1 (1 files); Downloading service1/service1; 16s remote, remote-cache
INFO: Found 4 targets...
INFO: Elapsed time: 62.137s, Critical Path: 53.13s
INFO: 10 processes: 1 internal, 9 remote.
INFO: Build completed successfully, 10 total actions
In Cargo, we can find this hash value as part of
OUT_DIR
value. Like./target/debug/build/mbedtls-664cfd9779753e4e/out/
.This hash value would be used to provide to
rustc
later like:LD_LIBRARY_PATH='/home/iraj/playground/target/debug/deps:/usr/lib' OUT_DIR=/home/iraj/playground/target/debug/build/mbedtls-platform-support-61808abfaac0885d/out rustc --crate-name mbedtls_platform_support --edition=2018 /home/iraj/.cargo/git/checkouts/rust-mbedtls-d7cb4be7371ceda7/b3a6f77/mbedtls-platform-support/src/lib.rs --error-format=json --json=diagnostic-rendered-ansi,artifacts,future-incompat --crate-type lib --emit=dep-info,metadata,link -C embed-bitcode=no -C debuginfo=2 --cfg 'feature="std"' -C metadata=7124dd98a051d608 ......
Since Bazel acts differently we need to generate a similar hash value to provide to
rustc
. Also, we detect the build system by comparing theOUT_DIR
value.To be able to build
rust-mbedtls
with Bazel we need more changes that will register in the following PRs and this PR is just focused on solving the CrateId Metadata Hash issue.