Disallow enabling features without their implied features
This commit is contained in:
parent
0b98a0c727
commit
8818c95528
@ -277,7 +277,7 @@ pub fn check_tied_features(
|
|||||||
/// Used to generate cfg variables and apply features
|
/// Used to generate cfg variables and apply features
|
||||||
/// Must express features in the way Rust understands them
|
/// Must express features in the way Rust understands them
|
||||||
pub fn target_features(sess: &Session, allow_unstable: bool) -> Vec<Symbol> {
|
pub fn target_features(sess: &Session, allow_unstable: bool) -> Vec<Symbol> {
|
||||||
let mut features = FxHashSet::default();
|
let mut features = vec![];
|
||||||
|
|
||||||
// Add base features for the target
|
// Add base features for the target
|
||||||
let target_machine = create_informational_target_machine(sess, true);
|
let target_machine = create_informational_target_machine(sess, true);
|
||||||
@ -313,7 +313,9 @@ pub fn target_features(sess: &Session, allow_unstable: bool) -> Vec<Symbol> {
|
|||||||
if enabled {
|
if enabled {
|
||||||
features.extend(sess.target.implied_target_features(std::iter::once(feature)));
|
features.extend(sess.target.implied_target_features(std::iter::once(feature)));
|
||||||
} else {
|
} else {
|
||||||
features.remove(&feature);
|
features.retain(|f| {
|
||||||
|
!sess.target.implied_target_features(std::iter::once(*f)).contains(&feature)
|
||||||
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
use rustc_ast::ast;
|
use rustc_ast::ast;
|
||||||
use rustc_attr::InstructionSetAttr;
|
use rustc_attr::InstructionSetAttr;
|
||||||
use rustc_data_structures::fx::FxIndexSet;
|
use rustc_data_structures::fx::FxIndexSet;
|
||||||
use rustc_data_structures::unord::{ExtendUnord, UnordMap, UnordSet};
|
use rustc_data_structures::unord::{UnordMap, UnordSet};
|
||||||
use rustc_errors::Applicability;
|
use rustc_errors::Applicability;
|
||||||
use rustc_hir::def::DefKind;
|
use rustc_hir::def::DefKind;
|
||||||
use rustc_hir::def_id::{DefId, LocalDefId, LOCAL_CRATE};
|
use rustc_hir::def_id::{DefId, LocalDefId, LOCAL_CRATE};
|
||||||
@ -108,8 +108,7 @@ pub fn from_target_feature(
|
|||||||
// Add implied features
|
// Add implied features
|
||||||
let mut implied_target_features = UnordSet::new();
|
let mut implied_target_features = UnordSet::new();
|
||||||
for feature in added_target_features.iter() {
|
for feature in added_target_features.iter() {
|
||||||
implied_target_features
|
implied_target_features.extend(tcx.implied_target_features(*feature).clone());
|
||||||
.extend_unord(tcx.implied_target_features(*feature).clone().into_items());
|
|
||||||
}
|
}
|
||||||
for feature in added_target_features.iter() {
|
for feature in added_target_features.iter() {
|
||||||
implied_target_features.remove(feature);
|
implied_target_features.remove(feature);
|
||||||
@ -179,7 +178,8 @@ pub(crate) fn provide(providers: &mut Providers) {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
implied_target_features: |tcx, feature| {
|
implied_target_features: |tcx, feature| {
|
||||||
tcx.sess.target.implied_target_features(std::iter::once(feature)).into()
|
UnordSet::from(tcx.sess.target.implied_target_features(std::iter::once(feature)))
|
||||||
|
.into_sorted_stable_ord()
|
||||||
},
|
},
|
||||||
asm_target_features,
|
asm_target_features,
|
||||||
..*providers
|
..*providers
|
||||||
|
@ -319,18 +319,12 @@ fn check_fn_target_features(&self, instance: ty::Instance<'tcx>) -> InterpResult
|
|||||||
.iter()
|
.iter()
|
||||||
.any(|feature| !self.tcx.sess.target_features.contains(&feature.name))
|
.any(|feature| !self.tcx.sess.target_features.contains(&feature.name))
|
||||||
{
|
{
|
||||||
// Don't include implicit features in the error, unless only implicit features are
|
|
||||||
// missing. This should be rare, because it can only happen when an implicit feature
|
|
||||||
// is disabled, e.g. `+avx2,-avx`
|
|
||||||
let missing_explicit_features = attrs.target_features.iter().any(|feature| {
|
|
||||||
!feature.implied && !self.tcx.sess.target_features.contains(&feature.name)
|
|
||||||
});
|
|
||||||
throw_ub_custom!(
|
throw_ub_custom!(
|
||||||
fluent::const_eval_unavailable_target_features_for_fn,
|
fluent::const_eval_unavailable_target_features_for_fn,
|
||||||
unavailable_feats = attrs
|
unavailable_feats = attrs
|
||||||
.target_features
|
.target_features
|
||||||
.iter()
|
.iter()
|
||||||
.filter(|&feature| !(missing_explicit_features && feature.implied)
|
.filter(|&feature| !feature.implied
|
||||||
&& !self.tcx.sess.target_features.contains(&feature.name))
|
&& !self.tcx.sess.target_features.contains(&feature.name))
|
||||||
.fold(String::new(), |mut s, feature| {
|
.fold(String::new(), |mut s, feature| {
|
||||||
if !s.is_empty() {
|
if !s.is_empty() {
|
||||||
|
@ -2183,7 +2183,7 @@
|
|||||||
desc { "looking up supported target features" }
|
desc { "looking up supported target features" }
|
||||||
}
|
}
|
||||||
|
|
||||||
query implied_target_features(feature: Symbol) -> &'tcx UnordSet<Symbol> {
|
query implied_target_features(feature: Symbol) -> &'tcx Vec<Symbol> {
|
||||||
arena_cache
|
arena_cache
|
||||||
eval_always
|
eval_always
|
||||||
desc { "looking up implied target features" }
|
desc { "looking up implied target features" }
|
||||||
|
@ -447,19 +447,11 @@ fn visit_expr(&mut self, expr: &'a Expr<'tcx>) {
|
|||||||
self.body_target_features.iter().any(|f| f.name == feature.name)
|
self.body_target_features.iter().any(|f| f.name == feature.name)
|
||||||
})
|
})
|
||||||
{
|
{
|
||||||
// Don't include implicit features in the error, unless only implicit
|
|
||||||
// features are missing.
|
|
||||||
let missing_explicit_features = callee_features.iter().any(|feature| {
|
|
||||||
!feature.implied
|
|
||||||
&& !self.body_target_features.iter().any(|body_feature| {
|
|
||||||
!feature.implied && body_feature.name == feature.name
|
|
||||||
})
|
|
||||||
});
|
|
||||||
let missing: Vec<_> = callee_features
|
let missing: Vec<_> = callee_features
|
||||||
.iter()
|
.iter()
|
||||||
.copied()
|
.copied()
|
||||||
.filter(|feature| {
|
.filter(|feature| {
|
||||||
!(missing_explicit_features && feature.implied)
|
!feature.implied
|
||||||
&& !self
|
&& !self
|
||||||
.body_target_features
|
.body_target_features
|
||||||
.iter()
|
.iter()
|
||||||
|
Loading…
Reference in New Issue
Block a user