Skip to content

Commit 674a7ad

Browse files
committed
Add an error when full metadata was not found
1 parent 4dca28c commit 674a7ad

File tree

3 files changed

+35
-1
lines changed

3 files changed

+35
-1
lines changed

compiler/rustc_metadata/messages.ftl

+4
Original file line numberDiff line numberDiff line change
@@ -97,6 +97,10 @@ metadata_found_staticlib =
9797
found staticlib `{$crate_name}` instead of rlib or dylib{$add_info}
9898
.help = please recompile that crate using --crate-type lib
9999
100+
metadata_full_metadata_not_found =
101+
only metadata stub found for `{$flavor}` dependency `{$crate_name}`
102+
please provide path to the corresponding .rmeta file with full metadata
103+
100104
metadata_global_alloc_required =
101105
no global memory allocator found but one is required; link to std or add `#[global_allocator]` to a static item that implements the GlobalAlloc trait
102106

compiler/rustc_metadata/src/errors.rs

+9
Original file line numberDiff line numberDiff line change
@@ -525,6 +525,15 @@ impl<G: EmissionGuarantee> Diagnostic<'_, G> for MultipleCandidates {
525525
}
526526
}
527527

528+
#[derive(Diagnostic)]
529+
#[diag(metadata_full_metadata_not_found)]
530+
pub(crate) struct FullMetadataNotFound {
531+
#[primary_span]
532+
pub span: Span,
533+
pub flavor: CrateFlavor,
534+
pub crate_name: Symbol,
535+
}
536+
528537
#[derive(Diagnostic)]
529538
#[diag(metadata_symbol_conflicts_current, code = E0519)]
530539
pub struct SymbolConflictsCurrent {

compiler/rustc_metadata/src/locator.rs

+22-1
Original file line numberDiff line numberDiff line change
@@ -654,7 +654,24 @@ impl<'a> CrateLocator<'a> {
654654
continue;
655655
}
656656
}
657-
*slot = Some((hash, metadata, lib.clone()));
657+
658+
// We error eagerly here. If we're locating a rlib, then in theory the full metadata
659+
// could still be in a (later resolved) dylib. In practice, if the rlib and dylib
660+
// were produced in a way where one has full metadata and the other hasn't, it would
661+
// mean that they were compiled using different compiler flags and probably also have
662+
// a different SVH value.
663+
if metadata.get_header().is_stub {
664+
// `is_stub` should never be true for .rmeta files.
665+
assert_ne!(flavor, CrateFlavor::Rmeta);
666+
667+
// Because rmeta files are resolved before rlib/dylib files, if this is a stub and
668+
// we haven't found a slot already, it means that the full metadata is missing.
669+
if slot.is_none() {
670+
return Err(CrateError::FullMetadataNotFound(self.crate_name, flavor));
671+
}
672+
} else {
673+
*slot = Some((hash, metadata, lib.clone()));
674+
}
658675
ret = Some((lib, kind));
659676
}
660677

@@ -916,6 +933,7 @@ pub(crate) enum CrateError {
916933
ExternLocationNotExist(Symbol, PathBuf),
917934
ExternLocationNotFile(Symbol, PathBuf),
918935
MultipleCandidates(Symbol, CrateFlavor, Vec<PathBuf>),
936+
FullMetadataNotFound(Symbol, CrateFlavor),
919937
SymbolConflictsCurrent(Symbol),
920938
StableCrateIdCollision(Symbol, Symbol),
921939
DlOpen(String, String),
@@ -966,6 +984,9 @@ impl CrateError {
966984
CrateError::MultipleCandidates(crate_name, flavor, candidates) => {
967985
dcx.emit_err(errors::MultipleCandidates { span, crate_name, flavor, candidates });
968986
}
987+
CrateError::FullMetadataNotFound(crate_name, flavor) => {
988+
dcx.emit_err(errors::FullMetadataNotFound { span, crate_name, flavor });
989+
}
969990
CrateError::SymbolConflictsCurrent(root_name) => {
970991
dcx.emit_err(errors::SymbolConflictsCurrent { span, crate_name: root_name });
971992
}

0 commit comments

Comments
 (0)