-
Notifications
You must be signed in to change notification settings - Fork 13.2k
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
Initial support for dynamically linked crates #134767
base: master
Are you sure you want to change the base?
Conversation
cc @m-ou-se |
This comment has been minimized.
This comment has been minimized.
I think that is indicative of a fundamental issue with your implementation. Adding new private impls should not change the ABI. The whole point of the RFC is to allow changing the implementation of the dylib without changing it's ABI. We already have support for dylibs where the ABI depends on the implementation. |
Also the interface file is missing either the Replacing all function bodies in an interface file with |
A stable ABI across rustc versions is a lot more involved than just mangling all symbols the same. You have to make sure multiple copies of the standard library seamlessly interoperate including using the same Edit: To put it another way, I did strongly prefer if stable ABI across rustc versions and stable ABI across crate versions with a single rustc version are treated as entirely separate problems implemented entirely separately. For stable ABI across crate versions you don't need to generate interface files. You can just reuse the existing rmeta file format, but only encode the subset of the items corresponding to the stable ABI. |
I can suggest the following solution for the above problem with impl's: Split indices (disambiguators) into 2 sets:
One of the main objectives of this proposal is to allow building the library and the application with different compiler versions. This requires either a metadata format compatible across rustc versions or some form of source code. Stable metadata format is not a part of the RFC.
"stable ABI" is a bad wording here. Neither I nor the RFC offers a stable ABI. And all these issues are outside the scope of the proposal. |
These two statements are conflicting with each other. Being able to build a library and application with a different compiler version requires both to share an ABI, in other words it requires having a stable ABI. |
Only |
This comment was marked as resolved.
This comment was marked as resolved.
This comment has been minimized.
This comment has been minimized.
This comment was marked as resolved.
This comment was marked as resolved.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment has been minimized.
This comment was marked as resolved.
This comment was marked as resolved.
Some changes occurred in compiler/rustc_codegen_gcc Some changes occurred in compiler/rustc_passes/src/check_attr.rs Some changes occurred in src/tools/rustfmt cc @rust-lang/rustfmt This PR modifies cc @jieyouxu These commits modify the If this was unintentional then you should revert the changes before this PR is merged. |
r? @bjorn3 (since you already did a bunch fo reviewers, or reroll) |
I can split it into several PR's/commits if necessary. |
@@ -1002,7 +1002,9 @@ impl<'a, 'tcx> Builder<'a, 'tcx> { | |||
self.source_scope = source_scope; | |||
} | |||
|
|||
if self.tcx.intrinsic(self.def_id).is_some_and(|i| i.must_be_overridden) { | |||
if self.tcx.intrinsic(self.def_id).is_some_and(|i| i.must_be_overridden) | |||
|| self.tcx.is_interface_build() |
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.
We should just skip building MIR entirely rather than adding a ad-hoc hack like this.
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 don't understand how to do this correctly. Many analysis passes depend on mir_built
. Adding checks around these passes seems like a bad idea.
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.
Maybe skip it in encode_mir
in metadata encoding?
Since we are in --emit=metadata
most MIR won't (?) be built unless it's needed for encoding.
@rustbot ready |
This comment was marked as resolved.
This comment was marked as resolved.
Ping @bjorn3 , since it has been 2 weeks. |
There are also several big, but non-blocking, issues here.
|
The PR description also needs some updates. |
Some changes occurred in compiler/rustc_codegen_ssa |
The job Click to see the possible cause of the failure (guessed by this bot)
|
.arg("--emit=metadata") | ||
.arg(format!("--crate-name={}", crate_name)) | ||
.arg(format!("--out-dir={}", tmp_path.path().display())) | ||
.arg("--crate-type=rlib") // shadow `sdylib` crate type in interface build |
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 think for interface builds the crate type needs to be overridden in fn collect_crate_types
, similarly to --test
builds.
Otherwise it doesn't affect even tests added in this PR, which set #![crate_type = "sdylib"]
through an attribute.
@@ -1488,6 +1488,17 @@ impl<'a> CrateMetadataRef<'a> { | |||
tcx.arena.alloc_from_iter(self.root.lang_items_missing.decode(self)) | |||
} | |||
|
|||
fn get_exportable_items(self) -> impl Iterator<Item = DefId> + 'a { |
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.
fn get_exportable_items(self) -> impl Iterator<Item = DefId> + 'a { | |
fn get_exportable_items(self) -> impl Iterator<Item = DefId> { |
IIUC, this should no longer be necessary on 2024 edition.
(Same in get_stable_order_of_exportable_impls
below.)
let exportable_items = stat!("exportable-items", || self.encode_exportable_items()); | ||
|
||
let stable_order_of_exportable_impls = | ||
stat!("exportable-items", || self.encode_stable_order_of_exportable_impls()); |
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.
Should these be encoded only in interface build mode?
@@ -506,6 +506,11 @@ pub static BUILTIN_ATTRIBUTES: &[BuiltinAttribute] = &[ | |||
naked_functions, experimental!(naked) | |||
), | |||
|
|||
gated!( | |||
export, Normal, template!(Word), WarnFollowing, | |||
EncodeCrossCrate::Yes, experimental!(export) |
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.
EncodeCrossCrate::Yes, experimental!(export) | |
EncodeCrossCrate::No, experimental!(export) |
The results of exportable_items
are encoded instead.
@@ -715,6 +720,27 @@ pub fn write_dep_info(tcx: TyCtxt<'_>) { | |||
} | |||
} | |||
|
|||
pub fn write_interface<'tcx>(tcx: TyCtxt<'tcx>) { | |||
if !tcx.crate_types().contains(&rustc_session::config::CrateType::Sdylib) |
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 is redundant now that interfaces are built as rlibs.
@@ -883,6 +909,10 @@ fn run_required_analyses(tcx: TyCtxt<'_>) { | |||
CStore::from_tcx(tcx).report_unused_deps(tcx); | |||
}, | |||
{ | |||
if tcx.crate_types().contains(&CrateType::Sdylib) { |
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 works accidentally because all the tests use #![crate_type = "sdylib"]
.
I think it's supposed to use if is_inteface_buid
instead.
The is_inteface_buid
optimization can be put inside the exportable_items
and stable_order_of_exportable_impls
queries, then we won't need to do it explicitly neither here nor in metadata encoder.
@@ -601,6 +614,11 @@ impl<'a> CrateLocator<'a> { | |||
} | |||
Err(MetadataError::LoadFailure(err)) => { | |||
info!("no metadata found: {}", err); | |||
// Metadata was loaded from interface file earlier. | |||
if let Some((_, _, _, CrateFlavor::SDylib)) = slot { |
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.
if let Some((_, _, _, CrateFlavor::SDylib)) = slot { | |
if let Some((.., CrateFlavor::SDylib)) = slot { |
@@ -762,7 +780,8 @@ impl<'a> CrateLocator<'a> { | |||
} | |||
|
|||
// Extract the dylib/rlib/rmeta triple. | |||
self.extract_lib(rlibs, rmetas, dylibs).map(|opt| opt.map(|(_, lib)| lib)) | |||
self.extract_lib(rlibs, rmetas, dylibs, FxIndexMap::default()) |
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.
--extern name=/path/to/interface.rs
is not implemented.
Ok(tmp_path) => tmp_path, | ||
Err(error) => { | ||
return Err(MetadataError::LoadFailure(format!( | ||
"Cannot create temporary directory: {}", |
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.
"Cannot create temporary directory: {}", | |
"couldn't create a temp dir: {}", |
To match FailedCreateTempdir
.
CrateFlavor::SDylib => { | ||
let compiler = std::env::current_exe().map_err(|_err| { | ||
MetadataError::LoadFailure( | ||
"couldn't obtain current compiler binary when loading `extern dyn` dependency" |
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.
Outdated message, extern dyn
no longer exist.
☔ The latest upstream changes (presumably #137535) made this pull request unmergeable. Please resolve the merge conflicts. |
This PR is an initial implementation of rust-lang/rfcs#3435 proposal.
component 1: interface generator
Interface generator - a tool for generating a stripped version of crate source code. The interface is like a C header with only "exported" items included, and function bodies are omitted. For example, initial crate:
generated interface:
In this example
bar
function is not a part of the interface as it doesn't have#[export]
attribute.To emit interface use a new
sdylib
crate type which is basically the same asdylib
, but it also produces the interface as a second artifact. The current interface name islib{crate_name}.rs
.Interface generator was implemented as part of the HIR pretty-printer. In order to filter out unnecessary items, the
PpAnn
trait was used.Why was it decided to use a design with an auto-generated interface?
One of the main objectives of this proposal is to allow building the library and the application with different compiler versions. This requires either a metadata format compatible across rustc versions or some form of a source code. The option with a stable metadata format was rejected because it is not a part of the RFC. (discussion)
Regarding the design with interfaces there are 2 possibilities: manually written or auto-generated. I would personally prefer the auto-generated interface for the following reason: we can put it in the dynamic library instead of metadata, which will make it completely invisible to users. (this was my initial plan, but since the PR is already big enough, I decided to postpone it)
But even if we end up with a different design, I believe the interface generator could be a useful tool for testing and experimenting with the entire feature.
component 2: crate loader
When building dynamic dependencies, the crate loader searches for the interface in the file system, builds the interface without codegen and loads it's metadata. For now, it's assumed that interface and dynamic lib are located in the same directory.
extern dyn crate
annotation serves as a signal for the building of a dynamic dependency.Here are the code and commands that corresponds to the compilation process:
P.S. The interface name/format and rules for file system routing can be changed further.
component 3: exportable items collector
Query for collecting exportable items. Which items are exportable is defined here .
component 4: "stable" mangling scheme
The mangling scheme proposed in the RFC consists of two parts: a mangled item path and a hash of the signature.
mangled item path
For the first part of the symbol it has been decided to reuse the
v0
mangling scheme as it is mostly independent of the compiler internals.The first exception is an impl's disambiguation. During symbol mangling rustc uses a special index to distinguish between two impls of the same type in the same module(See
DisambiguatedDefPathData
). The calculation of this index may depend on private items, but private items should not affect the ABI. Example:In order to make disambiguation independent of the compiler version we can assign an id to each impl according to their relative order in the source code. However, I have not found the right way to get this order, so I decided to use:
intravisit::Visitor
traversal.#[rustc_stable_impl_id]
that outputs this sequential number as a compiler error. If the visitor's implementation is changed, the corresponding test will fail. Then you will need to rewrite the implementation ofstable_order_of_exportable_impls
query to preserve order.P.S. is it possible to compare spans instead?
The second exception is
StableCrateId
which is used to disambiguate different crates.StableCrateId
consists of crate name,-Cmetadata
arguments and compiler version. At the moment, I have decided to keep only the crate name, but a more consistent approach to crate disambiguation could be added in the future.hash of the signature
Second part of the symbol name is 128 bit hash containing relevant type information. For now, it includes:
#[export(unsafe_stable_abi = "hash")]
syntax for ADT types with private fields is not yet implemented.