Skip to content

Commit 67a5171

Browse files
committed
Auto merge of #40008 - eddyb:lazy-12, r=nikomatsakis
[12/12] On-demand type-checking, const-evaluation, MIR building & const-qualification. _This is the last of a series ([prev](#38813)) of patches designed to rework rustc into an out-of-order on-demand pipeline model for both better feature support (e.g. [MIR-based](https://github.com/solson/miri) early constant evaluation) and incremental execution of compiler passes (e.g. type-checking), with beneficial consequences to IDE support as well. If any motivation is unclear, please ask for additional PR description clarifications or code comments._ <hr> As this contains all of the changes that didn't fit neatly into other PRs, I'll be elaborating a bit: ### User-facing changes * when determining whether an `impl Trait` type implements an auto-trait (e.g. `Send` or `Sync`), the function the `impl Trait` came from has to be inferred and type-checking, disallowing cycles * this results from not having an obvious place to put the "deferred obligation" in on-demand atm * while we could model side-effects like that and "post-processing passes" better, it's still more limiting than being able to know the result in the original function (e.g. specialization) *and* there are serious problems around region-checking (if a `Send` impl required `'static`, it wasn't enforced) * early const-eval requires type-checking and const-qualification to be performed first, which means: * you get the intended errors before (if any) constant evaluation error that is simply fallout * associated consts should always work now, and `const fn` type parameters are properly tracked * don't get too excited, array lengths still can't depend on type parameters * #38864 works as intended now, with `Self` being allowed in `impl` bounds * #32205 is largely improved, with associated types being limited to "exact match" `impl`s (as opposed to traversing the specialization graph to resolve unspecified type parameters to their defaults in another `impl` or in the `trait`) *while* checking for overlaps building the specialization graph for that trait - once all the trait impls' have been checked for coherence (including ahead-of-time/on-demand), it's uniform * [crater report](https://gist.github.com/eddyb/bbb869072468c7e08d6d808e75938051) looks clean (aside from `clippy` which broke due to `rustc` internal changes) ### Compiler-internal changes * `ty::Generics` * no longer contains the actual type parameter defaults, instead they're associated with the type parameter's `DefId`, like associated types in a trait definition * this allows computing `ty::Generics` as a leaf (reading only its own HIR) * holds a mapping from `DefIndex` of type parameters to their indices * `ty::AdtDef` * only tracks `#[repr(simd)]` in its `ReprOptions` `repr` field * doesn't contain `enum` discriminant values, but instead each variant either refers to either an explicit value for its discriminant, or the distance from the last explicit discriminant, if any * the `.discriminants(tcx)` method produces an iterator of `ConstInt` values, looking up explicit discriminants in a separate map, if necessary * this allows computing `ty::AdtDef` as a leaf (reading only its own HIR) * Small note: the two above (`Generics`, `AdtDef`), `TraitDef` and `AssociatedItem` should probably end up as part of the HIR, eventually, as they're trivially constructed from it * `ty::FnSig` * now also holds ABI and unsafety, alongside argument types, return type and C variadicity * `&ty::BareFnTy` and `ty::ClosureTy` have been replaced with `PolyFnSig = Binder<FnSig>` * `BareFnTy` was interned and `ClosureTy` was treated as non-trivial to `Clone` because they had a `PolyFnSig` and so used to contain a `Vec<Ty>` (now `&[Ty]`) * `ty::maps` * all the `DepTrackingMap`s have been grouped in a structure available at `tcx.maps` * when creating the `tcx`, a set of `Providers` (one `fn` pointer per map) is required for the local crate, and one for all other crates (i.e. metadata loading), `librustc_driver` plugging the various crates (e.g. `librustc_metadata`, `librustc_typeck`, `librustc_mir`) into it * when a map is queried and the value is missing, the appropriate `fn` pointer from the `Providers` of that crate is called with the `TyCtxt` and the key being queried, to produce the value on-demand * `rustc_const_eval` * demands both `typeck_tables` and `mir_const_qualif` (in preparation for miri) * tracks `Substs` in `ConstVal::Function` for `const fn` calls * returns `TypeckError` if type-checking has failed (or cases that can only be reached if it had) * this error kind is never reported, resulting in less noisy/redundant diagnostics * fixes #39548 (testcase by @larsluthman, taken from #39812, which this supersedes) * on-demand has so far been hooked up to: * `rustc_metadata::cstore_impl`: `ty`, `generics`, `predicates`, `super_predicates`, `trait_def`, `adt_def`, `variances`, `associated_item_def_ids`, `associated_item`, `impl_trait_ref`, `custom_coerce_unsized_kind`, `mir`, `mir_const_qualif`, `typeck_tables`, `closure_kind`, `closure_type` * `rustc_typeck::collect`: `ty`, `generics`, `predicates`, `super_predicates`, `type_param_predicates`, `trait_def`, `adt_def`, `impl_trait_ref` * `rustc_typeck::coherence`: `coherent_trait`, `coherent_inherent_impls` * `rustc_typeck::check`: `typeck_tables`, `closure_type`, `closure_kind` * `rustc_mir::mir_map`: `mir` * `rustc_mir::transform::qualify_consts`: `mir_const_qualif`
2 parents 808dbcc + d9f0a94 commit 67a5171

File tree

207 files changed

+4880
-5895
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

207 files changed

+4880
-5895
lines changed

src/librustc/dep_graph/dep_node.rs

+8
Original file line numberDiff line numberDiff line change
@@ -70,6 +70,7 @@ pub enum DepNode<D: Clone + Debug> {
7070
Resolve,
7171
EntryPoint,
7272
CheckEntryFn,
73+
CoherenceCheckTrait(D),
7374
CoherenceCheckImpl(D),
7475
CoherenceOverlapCheck(D),
7576
CoherenceOverlapCheckSpecial(D),
@@ -109,11 +110,13 @@ pub enum DepNode<D: Clone + Debug> {
109110
// predicates for an item wind up in `ItemSignature`).
110111
AssociatedItems(D),
111112
ItemSignature(D),
113+
TypeParamPredicates((D, D)),
112114
SizedConstraint(D),
113115
AssociatedItemDefIds(D),
114116
InherentImpls(D),
115117
TypeckTables(D),
116118
UsedTraitImports(D),
119+
MonomorphicConstEval(D),
117120

118121
// The set of impls for a given trait. Ultimately, it would be
119122
// nice to get more fine-grained here (e.g., to include a
@@ -239,6 +242,7 @@ impl<D: Clone + Debug> DepNode<D> {
239242
MetaData(ref d) => op(d).map(MetaData),
240243
CollectItem(ref d) => op(d).map(CollectItem),
241244
CollectItemSig(ref d) => op(d).map(CollectItemSig),
245+
CoherenceCheckTrait(ref d) => op(d).map(CoherenceCheckTrait),
242246
CoherenceCheckImpl(ref d) => op(d).map(CoherenceCheckImpl),
243247
CoherenceOverlapCheck(ref d) => op(d).map(CoherenceOverlapCheck),
244248
CoherenceOverlapCheckSpecial(ref d) => op(d).map(CoherenceOverlapCheckSpecial),
@@ -258,11 +262,15 @@ impl<D: Clone + Debug> DepNode<D> {
258262
TransInlinedItem(ref d) => op(d).map(TransInlinedItem),
259263
AssociatedItems(ref d) => op(d).map(AssociatedItems),
260264
ItemSignature(ref d) => op(d).map(ItemSignature),
265+
TypeParamPredicates((ref item, ref param)) => {
266+
Some(TypeParamPredicates((try_opt!(op(item)), try_opt!(op(param)))))
267+
}
261268
SizedConstraint(ref d) => op(d).map(SizedConstraint),
262269
AssociatedItemDefIds(ref d) => op(d).map(AssociatedItemDefIds),
263270
InherentImpls(ref d) => op(d).map(InherentImpls),
264271
TypeckTables(ref d) => op(d).map(TypeckTables),
265272
UsedTraitImports(ref d) => op(d).map(UsedTraitImports),
273+
MonomorphicConstEval(ref d) => op(d).map(MonomorphicConstEval),
266274
TraitImpls(ref d) => op(d).map(TraitImpls),
267275
TraitItems(ref d) => op(d).map(TraitItems),
268276
ReprHints(ref d) => op(d).map(ReprHints),

src/librustc/dep_graph/dep_tracking_map.rs

+6
Original file line numberDiff line numberDiff line change
@@ -11,6 +11,7 @@
1111
use hir::def_id::DefId;
1212
use rustc_data_structures::fx::FxHashMap;
1313
use std::cell::RefCell;
14+
use std::collections::hash_map::Entry;
1415
use std::ops::Index;
1516
use std::hash::Hash;
1617
use std::marker::PhantomData;
@@ -67,6 +68,11 @@ impl<M: DepTrackingMapConfig> DepTrackingMap<M> {
6768
assert!(old_value.is_none());
6869
}
6970

71+
pub fn entry(&mut self, k: M::Key) -> Entry<M::Key, M::Value> {
72+
self.write(&k);
73+
self.map.entry(k)
74+
}
75+
7076
pub fn contains_key(&self, k: &M::Key) -> bool {
7177
self.read(k);
7278
self.map.contains_key(k)

src/librustc/diagnostics.rs

+17-73
Original file line numberDiff line numberDiff line change
@@ -390,44 +390,6 @@ RFC. It is, however, [currently unimplemented][iss15872].
390390
[iss15872]: https://github.com/rust-lang/rust/issues/15872
391391
"##,
392392

393-
E0109: r##"
394-
You tried to give a type parameter to a type which doesn't need it. Erroneous
395-
code example:
396-
397-
```compile_fail,E0109
398-
type X = u32<i32>; // error: type parameters are not allowed on this type
399-
```
400-
401-
Please check that you used the correct type and recheck its definition. Perhaps
402-
it doesn't need the type parameter.
403-
404-
Example:
405-
406-
```
407-
type X = u32; // this compiles
408-
```
409-
410-
Note that type parameters for enum-variant constructors go after the variant,
411-
not after the enum (Option::None::<u32>, not Option::<u32>::None).
412-
"##,
413-
414-
E0110: r##"
415-
You tried to give a lifetime parameter to a type which doesn't need it.
416-
Erroneous code example:
417-
418-
```compile_fail,E0110
419-
type X = u32<'static>; // error: lifetime parameters are not allowed on
420-
// this type
421-
```
422-
423-
Please check that the correct type was used and recheck its definition; perhaps
424-
it doesn't need the lifetime parameter. Example:
425-
426-
```
427-
type X = u32; // ok!
428-
```
429-
"##,
430-
431393
E0133: r##"
432394
Unsafe code was used outside of an unsafe function or block.
433395
@@ -627,41 +589,6 @@ attributes:
627589
See also https://doc.rust-lang.org/book/no-stdlib.html
628590
"##,
629591

630-
E0229: r##"
631-
An associated type binding was done outside of the type parameter declaration
632-
and `where` clause. Erroneous code example:
633-
634-
```compile_fail,E0229
635-
pub trait Foo {
636-
type A;
637-
fn boo(&self) -> <Self as Foo>::A;
638-
}
639-
640-
struct Bar;
641-
642-
impl Foo for isize {
643-
type A = usize;
644-
fn boo(&self) -> usize { 42 }
645-
}
646-
647-
fn baz<I>(x: &<I as Foo<A=Bar>>::A) {}
648-
// error: associated type bindings are not allowed here
649-
```
650-
651-
To solve this error, please move the type bindings in the type parameter
652-
declaration:
653-
654-
```ignore
655-
fn baz<I: Foo<A=Bar>>(x: &<I as Foo>::A) {} // ok!
656-
```
657-
658-
Or in the `where` clause:
659-
660-
```ignore
661-
fn baz<I>(x: &<I as Foo>::A) where I: Foo<A=Bar> {}
662-
```
663-
"##,
664-
665592
E0261: r##"
666593
When using a lifetime like `'a` in a type, it must be declared before being
667594
used.
@@ -1390,6 +1317,23 @@ error. To resolve it, add an `else` block having the same type as the `if`
13901317
block.
13911318
"##,
13921319

1320+
E0391: r##"
1321+
This error indicates that some types or traits depend on each other
1322+
and therefore cannot be constructed.
1323+
1324+
The following example contains a circular dependency between two traits:
1325+
1326+
```compile_fail,E0391
1327+
trait FirstTrait : SecondTrait {
1328+
1329+
}
1330+
1331+
trait SecondTrait : FirstTrait {
1332+
1333+
}
1334+
```
1335+
"##,
1336+
13931337
E0398: r##"
13941338
In Rust 1.3, the default object lifetime bounds are expected to change, as
13951339
described in RFC #1156 [1]. You are getting a warning because the compiler

src/librustc/hir/lowering.rs

+22-2
Original file line numberDiff line numberDiff line change
@@ -80,6 +80,9 @@ pub struct LoweringContext<'a> {
8080
impl_items: BTreeMap<hir::ImplItemId, hir::ImplItem>,
8181
bodies: FxHashMap<hir::BodyId, hir::Body>,
8282

83+
trait_impls: BTreeMap<DefId, Vec<NodeId>>,
84+
trait_default_impl: BTreeMap<DefId, NodeId>,
85+
8386
loop_scopes: Vec<NodeId>,
8487
is_in_loop_condition: bool,
8588

@@ -116,6 +119,8 @@ pub fn lower_crate(sess: &Session,
116119
trait_items: BTreeMap::new(),
117120
impl_items: BTreeMap::new(),
118121
bodies: FxHashMap(),
122+
trait_impls: BTreeMap::new(),
123+
trait_default_impl: BTreeMap::new(),
119124
loop_scopes: Vec::new(),
120125
is_in_loop_condition: false,
121126
type_def_lifetime_params: DefIdMap(),
@@ -201,6 +206,8 @@ impl<'a> LoweringContext<'a> {
201206
trait_items: self.trait_items,
202207
impl_items: self.impl_items,
203208
bodies: self.bodies,
209+
trait_impls: self.trait_impls,
210+
trait_default_impl: self.trait_default_impl,
204211
}
205212
}
206213

@@ -525,7 +532,7 @@ impl<'a> LoweringContext<'a> {
525532
return n;
526533
}
527534
assert!(!def_id.is_local());
528-
let (n, _) = self.sess.cstore.item_generics_own_param_counts(def_id);
535+
let n = self.sess.cstore.item_generics_cloned(def_id).regions.len();
529536
self.type_def_lifetime_params.insert(def_id, n);
530537
n
531538
});
@@ -1089,14 +1096,27 @@ impl<'a> LoweringContext<'a> {
10891096
hir::ItemUnion(vdata, self.lower_generics(generics))
10901097
}
10911098
ItemKind::DefaultImpl(unsafety, ref trait_ref) => {
1099+
let trait_ref = self.lower_trait_ref(trait_ref);
1100+
1101+
if let Def::Trait(def_id) = trait_ref.path.def {
1102+
self.trait_default_impl.insert(def_id, id);
1103+
}
1104+
10921105
hir::ItemDefaultImpl(self.lower_unsafety(unsafety),
1093-
self.lower_trait_ref(trait_ref))
1106+
trait_ref)
10941107
}
10951108
ItemKind::Impl(unsafety, polarity, ref generics, ref ifce, ref ty, ref impl_items) => {
10961109
let new_impl_items = impl_items.iter()
10971110
.map(|item| self.lower_impl_item_ref(item))
10981111
.collect();
10991112
let ifce = ifce.as_ref().map(|trait_ref| self.lower_trait_ref(trait_ref));
1113+
1114+
if let Some(ref trait_ref) = ifce {
1115+
if let Def::Trait(def_id) = trait_ref.path.def {
1116+
self.trait_impls.entry(def_id).or_insert(vec![]).push(id);
1117+
}
1118+
}
1119+
11001120
hir::ItemImpl(self.lower_unsafety(unsafety),
11011121
self.lower_impl_polarity(polarity),
11021122
self.lower_generics(generics),

src/librustc/hir/map/mod.rs

+44
Original file line numberDiff line numberDiff line change
@@ -437,6 +437,50 @@ impl<'hir> Map<'hir> {
437437
self.local_def_id(self.body_owner(id))
438438
}
439439

440+
pub fn ty_param_owner(&self, id: NodeId) -> NodeId {
441+
match self.get(id) {
442+
NodeItem(&Item { node: ItemTrait(..), .. }) => id,
443+
NodeTyParam(_) => self.get_parent_node(id),
444+
_ => {
445+
bug!("ty_param_owner: {} not a type parameter",
446+
self.node_to_string(id))
447+
}
448+
}
449+
}
450+
451+
pub fn ty_param_name(&self, id: NodeId) -> Name {
452+
match self.get(id) {
453+
NodeItem(&Item { node: ItemTrait(..), .. }) => {
454+
keywords::SelfType.name()
455+
}
456+
NodeTyParam(tp) => tp.name,
457+
_ => {
458+
bug!("ty_param_name: {} not a type parameter",
459+
self.node_to_string(id))
460+
}
461+
}
462+
}
463+
464+
pub fn trait_impls(&self, trait_did: DefId) -> &'hir [NodeId] {
465+
self.dep_graph.read(DepNode::TraitImpls(trait_did));
466+
467+
// NB: intentionally bypass `self.forest.krate()` so that we
468+
// do not trigger a read of the whole krate here
469+
self.forest.krate.trait_impls.get(&trait_did).map_or(&[], |xs| &xs[..])
470+
}
471+
472+
pub fn trait_default_impl(&self, trait_did: DefId) -> Option<NodeId> {
473+
self.dep_graph.read(DepNode::TraitImpls(trait_did));
474+
475+
// NB: intentionally bypass `self.forest.krate()` so that we
476+
// do not trigger a read of the whole krate here
477+
self.forest.krate.trait_default_impl.get(&trait_did).cloned()
478+
}
479+
480+
pub fn trait_is_auto(&self, trait_did: DefId) -> bool {
481+
self.trait_default_impl(trait_did).is_some()
482+
}
483+
440484
/// Get the attributes on the krate. This is preferable to
441485
/// invoking `krate.attrs` because it registers a tighter
442486
/// dep-graph access.

src/librustc/hir/mod.rs

+3
Original file line numberDiff line numberDiff line change
@@ -410,6 +410,9 @@ pub struct Crate {
410410
pub trait_items: BTreeMap<TraitItemId, TraitItem>,
411411
pub impl_items: BTreeMap<ImplItemId, ImplItem>,
412412
pub bodies: FxHashMap<BodyId, Body>,
413+
414+
pub trait_impls: BTreeMap<DefId, Vec<NodeId>>,
415+
pub trait_default_impl: BTreeMap<DefId, NodeId>,
413416
}
414417

415418
impl Crate {

src/librustc/infer/mod.rs

+15-16
Original file line numberDiff line numberDiff line change
@@ -505,7 +505,7 @@ impl<'a, 'gcx, 'tcx> TyCtxt<'a, 'gcx, 'gcx> {
505505
evaluation_cache: traits::EvaluationCache::new(),
506506
projection_cache: RefCell::new(traits::ProjectionCache::new()),
507507
reported_trait_errors: RefCell::new(FxHashSet()),
508-
projection_mode: Reveal::NotSpecializable,
508+
projection_mode: Reveal::UserFacing,
509509
tainted_by_errors_flag: Cell::new(false),
510510
err_count_on_creation: self.sess.err_count(),
511511
obligations_in_snapshot: Cell::new(false),
@@ -600,7 +600,7 @@ impl_trans_normalize!('gcx,
600600
Ty<'gcx>,
601601
&'gcx Substs<'gcx>,
602602
ty::FnSig<'gcx>,
603-
&'gcx ty::BareFnTy<'gcx>,
603+
ty::PolyFnSig<'gcx>,
604604
ty::ClosureSubsts<'gcx>,
605605
ty::PolyTraitRef<'gcx>,
606606
ty::ExistentialTraitRef<'gcx>
@@ -1197,16 +1197,19 @@ impl<'a, 'gcx, 'tcx> InferCtxt<'a, 'gcx, 'tcx> {
11971197
/// as the substitutions for the default, `(T, U)`.
11981198
pub fn type_var_for_def(&self,
11991199
span: Span,
1200-
def: &ty::TypeParameterDef<'tcx>,
1200+
def: &ty::TypeParameterDef,
12011201
substs: &[Kind<'tcx>])
12021202
-> Ty<'tcx> {
1203-
let default = def.default.map(|default| {
1204-
type_variable::Default {
1203+
let default = if def.has_default {
1204+
let default = self.tcx.item_type(def.def_id);
1205+
Some(type_variable::Default {
12051206
ty: default.subst_spanned(self.tcx, substs, Some(span)),
12061207
origin_span: span,
1207-
def_id: def.default_def_id
1208-
}
1209-
});
1208+
def_id: def.def_id
1209+
})
1210+
} else {
1211+
None
1212+
};
12101213

12111214

12121215
let ty_var_id = self.type_variables
@@ -1646,20 +1649,16 @@ impl<'a, 'gcx, 'tcx> InferCtxt<'a, 'gcx, 'tcx> {
16461649
Some(self.tcx.closure_kind(def_id))
16471650
}
16481651

1649-
pub fn closure_type(&self,
1650-
def_id: DefId,
1651-
substs: ty::ClosureSubsts<'tcx>)
1652-
-> ty::ClosureTy<'tcx>
1653-
{
1652+
pub fn closure_type(&self, def_id: DefId) -> ty::PolyFnSig<'tcx> {
16541653
if let InferTables::InProgress(tables) = self.tables {
16551654
if let Some(id) = self.tcx.hir.as_local_node_id(def_id) {
1656-
if let Some(ty) = tables.borrow().closure_tys.get(&id) {
1657-
return ty.subst(self.tcx, substs.substs);
1655+
if let Some(&ty) = tables.borrow().closure_tys.get(&id) {
1656+
return ty;
16581657
}
16591658
}
16601659
}
16611660

1662-
self.tcx.closure_type(def_id, substs)
1661+
self.tcx.closure_type(def_id)
16631662
}
16641663
}
16651664

src/librustc/lib.rs

+1-1
Original file line numberDiff line numberDiff line change
@@ -39,6 +39,7 @@
3939
#![feature(rustc_diagnostic_macros)]
4040
#![feature(rustc_private)]
4141
#![feature(slice_patterns)]
42+
#![feature(specialization)]
4243
#![feature(staged_api)]
4344
#![feature(unboxed_closures)]
4445

@@ -75,7 +76,6 @@ pub mod infer;
7576
pub mod lint;
7677

7778
pub mod middle {
78-
pub mod astconv_util;
7979
pub mod expr_use_visitor;
8080
pub mod const_val;
8181
pub mod cstore;

0 commit comments

Comments
 (0)