rustc_hir_analysis/variance/
constraints.rs
1use hir::def_id::{DefId, LocalDefId};
7use rustc_hir as hir;
8use rustc_hir::def::DefKind;
9use rustc_middle::ty::{self, GenericArgKind, GenericArgsRef, Ty, TyCtxt};
10use rustc_middle::{bug, span_bug};
11use tracing::{debug, instrument};
12
13use super::terms::VarianceTerm::*;
14use super::terms::*;
15
16pub(crate) struct ConstraintContext<'a, 'tcx> {
17 pub terms_cx: TermsContext<'a, 'tcx>,
18
19 covariant: VarianceTermPtr<'a>,
21 contravariant: VarianceTermPtr<'a>,
22 invariant: VarianceTermPtr<'a>,
23 bivariant: VarianceTermPtr<'a>,
24
25 pub constraints: Vec<Constraint<'a>>,
26}
27
28#[derive(Copy, Clone)]
31pub(crate) struct Constraint<'a> {
32 pub inferred: InferredIndex,
33 pub variance: &'a VarianceTerm<'a>,
34}
35
36struct CurrentItem {
46 inferred_start: InferredIndex,
47}
48
49pub(crate) fn add_constraints_from_crate<'a, 'tcx>(
50 terms_cx: TermsContext<'a, 'tcx>,
51) -> ConstraintContext<'a, 'tcx> {
52 let tcx = terms_cx.tcx;
53 let covariant = terms_cx.arena.alloc(ConstantTerm(ty::Covariant));
54 let contravariant = terms_cx.arena.alloc(ConstantTerm(ty::Contravariant));
55 let invariant = terms_cx.arena.alloc(ConstantTerm(ty::Invariant));
56 let bivariant = terms_cx.arena.alloc(ConstantTerm(ty::Bivariant));
57 let mut constraint_cx = ConstraintContext {
58 terms_cx,
59 covariant,
60 contravariant,
61 invariant,
62 bivariant,
63 constraints: Vec::new(),
64 };
65
66 let crate_items = tcx.hir_crate_items(());
67
68 for def_id in crate_items.definitions() {
69 let def_kind = tcx.def_kind(def_id);
70 match def_kind {
71 DefKind::Struct | DefKind::Union | DefKind::Enum => {
72 constraint_cx.build_constraints_for_item(def_id);
73
74 let adt = tcx.adt_def(def_id);
75 for variant in adt.variants() {
76 if let Some(ctor_def_id) = variant.ctor_def_id() {
77 constraint_cx.build_constraints_for_item(ctor_def_id.expect_local());
78 }
79 }
80 }
81 DefKind::Fn | DefKind::AssocFn => constraint_cx.build_constraints_for_item(def_id),
82 DefKind::TyAlias if tcx.type_alias_is_lazy(def_id) => {
83 constraint_cx.build_constraints_for_item(def_id)
84 }
85 _ => {}
86 }
87 }
88
89 constraint_cx
90}
91
92impl<'a, 'tcx> ConstraintContext<'a, 'tcx> {
93 fn tcx(&self) -> TyCtxt<'tcx> {
94 self.terms_cx.tcx
95 }
96
97 fn build_constraints_for_item(&mut self, def_id: LocalDefId) {
98 let tcx = self.tcx();
99 debug!("build_constraints_for_item({})", tcx.def_path_str(def_id));
100
101 if tcx.generics_of(def_id).is_empty() {
103 return;
104 }
105
106 let inferred_start = self.terms_cx.inferred_starts[&def_id];
107 let current_item = &CurrentItem { inferred_start };
108 let ty = tcx.type_of(def_id).instantiate_identity();
109
110 if let DefKind::TyAlias = tcx.def_kind(def_id)
113 && tcx.type_alias_is_lazy(def_id)
114 {
115 self.add_constraints_from_ty(current_item, ty, self.covariant);
116 return;
117 }
118
119 match ty.kind() {
120 ty::Adt(def, _) => {
121 for field in def.all_fields() {
128 self.add_constraints_from_ty(
129 current_item,
130 tcx.type_of(field.did).instantiate_identity(),
131 self.covariant,
132 );
133 }
134 }
135
136 ty::FnDef(..) => {
137 self.add_constraints_from_sig(
138 current_item,
139 tcx.fn_sig(def_id).instantiate_identity(),
140 self.covariant,
141 );
142 }
143
144 ty::Error(_) => {}
145
146 _ => {
147 span_bug!(
148 tcx.def_span(def_id),
149 "`build_constraints_for_item` unsupported for this item"
150 );
151 }
152 }
153 }
154
155 fn add_constraint(&mut self, current: &CurrentItem, index: u32, variance: VarianceTermPtr<'a>) {
156 debug!("add_constraint(index={}, variance={:?})", index, variance);
157 self.constraints.push(Constraint {
158 inferred: InferredIndex(current.inferred_start.0 + index as usize),
159 variance,
160 });
161 }
162
163 fn contravariant(&mut self, variance: VarianceTermPtr<'a>) -> VarianceTermPtr<'a> {
164 self.xform(variance, self.contravariant)
165 }
166
167 fn invariant(&mut self, variance: VarianceTermPtr<'a>) -> VarianceTermPtr<'a> {
168 self.xform(variance, self.invariant)
169 }
170
171 fn constant_term(&self, v: ty::Variance) -> VarianceTermPtr<'a> {
172 match v {
173 ty::Covariant => self.covariant,
174 ty::Invariant => self.invariant,
175 ty::Contravariant => self.contravariant,
176 ty::Bivariant => self.bivariant,
177 }
178 }
179
180 fn xform(&mut self, v1: VarianceTermPtr<'a>, v2: VarianceTermPtr<'a>) -> VarianceTermPtr<'a> {
181 match (*v1, *v2) {
182 (_, ConstantTerm(ty::Covariant)) => {
183 v1
185 }
186
187 (ConstantTerm(c1), ConstantTerm(c2)) => self.constant_term(c1.xform(c2)),
188
189 _ => &*self.terms_cx.arena.alloc(TransformTerm(v1, v2)),
190 }
191 }
192
193 #[instrument(level = "debug", skip(self, current))]
194 fn add_constraints_from_invariant_args(
195 &mut self,
196 current: &CurrentItem,
197 args: GenericArgsRef<'tcx>,
198 variance: VarianceTermPtr<'a>,
199 ) {
200 let variance_i = self.invariant(variance);
202
203 for k in args {
204 match k.unpack() {
205 GenericArgKind::Lifetime(lt) => {
206 self.add_constraints_from_region(current, lt, variance_i)
207 }
208 GenericArgKind::Type(ty) => self.add_constraints_from_ty(current, ty, variance_i),
209 GenericArgKind::Const(val) => {
210 self.add_constraints_from_const(current, val, variance_i)
211 }
212 }
213 }
214 }
215
216 fn add_constraints_from_ty(
220 &mut self,
221 current: &CurrentItem,
222 ty: Ty<'tcx>,
223 variance: VarianceTermPtr<'a>,
224 ) {
225 debug!("add_constraints_from_ty(ty={:?}, variance={:?})", ty, variance);
226
227 match *ty.kind() {
228 ty::Bool
229 | ty::Char
230 | ty::Int(_)
231 | ty::Uint(_)
232 | ty::Float(_)
233 | ty::Str
234 | ty::Never
235 | ty::Foreign(..) => {
236 }
238
239 ty::FnDef(..) | ty::Coroutine(..) | ty::Closure(..) | ty::CoroutineClosure(..) => {
240 bug!("Unexpected unnameable type in variance computation: {ty}");
241 }
242
243 ty::Ref(region, ty, mutbl) => {
244 self.add_constraints_from_region(current, region, variance);
245 self.add_constraints_from_mt(current, &ty::TypeAndMut { ty, mutbl }, variance);
246 }
247
248 ty::Array(typ, len) => {
249 self.add_constraints_from_const(current, len, variance);
250 self.add_constraints_from_ty(current, typ, variance);
251 }
252
253 ty::Pat(typ, pat) => {
254 match *pat {
255 ty::PatternKind::Range { start, end } => {
256 self.add_constraints_from_const(current, start, variance);
257 self.add_constraints_from_const(current, end, variance);
258 }
259 }
260 self.add_constraints_from_ty(current, typ, variance);
261 }
262
263 ty::Slice(typ) => {
264 self.add_constraints_from_ty(current, typ, variance);
265 }
266
267 ty::RawPtr(ty, mutbl) => {
268 self.add_constraints_from_mt(current, &ty::TypeAndMut { ty, mutbl }, variance);
269 }
270
271 ty::Tuple(subtys) => {
272 for subty in subtys {
273 self.add_constraints_from_ty(current, subty, variance);
274 }
275 }
276
277 ty::Adt(def, args) => {
278 self.add_constraints_from_args(current, def.did(), args, variance);
279 }
280
281 ty::Alias(ty::Projection | ty::Inherent | ty::Opaque, ref data) => {
282 self.add_constraints_from_invariant_args(current, data.args, variance);
283 }
284
285 ty::Alias(ty::Weak, ref data) => {
286 self.add_constraints_from_args(current, data.def_id, data.args, variance);
287 }
288
289 ty::Dynamic(data, r, _) => {
290 self.add_constraints_from_region(current, r, variance);
292
293 if let Some(poly_trait_ref) = data.principal() {
294 self.add_constraints_from_invariant_args(
295 current,
296 poly_trait_ref.skip_binder().args,
297 variance,
298 );
299 }
300
301 for projection in data.projection_bounds() {
302 match projection.skip_binder().term.unpack() {
303 ty::TermKind::Ty(ty) => {
304 self.add_constraints_from_ty(current, ty, self.invariant);
305 }
306 ty::TermKind::Const(c) => {
307 self.add_constraints_from_const(current, c, self.invariant)
308 }
309 }
310 }
311 }
312
313 ty::Param(ref data) => {
314 self.add_constraint(current, data.index, variance);
315 }
316
317 ty::FnPtr(sig_tys, hdr) => {
318 self.add_constraints_from_sig(current, sig_tys.with(hdr), variance);
319 }
320
321 ty::UnsafeBinder(ty) => {
322 self.add_constraints_from_ty(current, ty.skip_binder(), variance);
324 }
325
326 ty::Error(_) => {
327 }
330
331 ty::Placeholder(..) | ty::CoroutineWitness(..) | ty::Bound(..) | ty::Infer(..) => {
332 bug!("unexpected type encountered in variance inference: {}", ty);
333 }
334 }
335 }
336
337 fn add_constraints_from_args(
340 &mut self,
341 current: &CurrentItem,
342 def_id: DefId,
343 args: GenericArgsRef<'tcx>,
344 variance: VarianceTermPtr<'a>,
345 ) {
346 debug!(
347 "add_constraints_from_args(def_id={:?}, args={:?}, variance={:?})",
348 def_id, args, variance
349 );
350
351 if args.is_empty() {
353 return;
354 }
355
356 let (local, remote) = if let Some(def_id) = def_id.as_local() {
357 (Some(self.terms_cx.inferred_starts[&def_id]), None)
358 } else {
359 (None, Some(self.tcx().variances_of(def_id)))
360 };
361
362 for (i, k) in args.iter().enumerate() {
363 let variance_decl = if let Some(InferredIndex(start)) = local {
364 self.terms_cx.inferred_terms[start + i]
368 } else {
369 self.constant_term(remote.as_ref().unwrap()[i])
372 };
373 let variance_i = self.xform(variance, variance_decl);
374 debug!(
375 "add_constraints_from_args: variance_decl={:?} variance_i={:?}",
376 variance_decl, variance_i
377 );
378 match k.unpack() {
379 GenericArgKind::Lifetime(lt) => {
380 self.add_constraints_from_region(current, lt, variance_i)
381 }
382 GenericArgKind::Type(ty) => self.add_constraints_from_ty(current, ty, variance_i),
383 GenericArgKind::Const(val) => {
384 self.add_constraints_from_const(current, val, variance)
385 }
386 }
387 }
388 }
389
390 fn add_constraints_from_const(
393 &mut self,
394 current: &CurrentItem,
395 c: ty::Const<'tcx>,
396 variance: VarianceTermPtr<'a>,
397 ) {
398 debug!("add_constraints_from_const(c={:?}, variance={:?})", c, variance);
399
400 match &c.kind() {
401 ty::ConstKind::Unevaluated(uv) => {
402 self.add_constraints_from_invariant_args(current, uv.args, variance);
403 }
404 _ => {}
405 }
406 }
407
408 fn add_constraints_from_sig(
411 &mut self,
412 current: &CurrentItem,
413 sig: ty::PolyFnSig<'tcx>,
414 variance: VarianceTermPtr<'a>,
415 ) {
416 let contra = self.contravariant(variance);
417 for &input in sig.skip_binder().inputs() {
418 self.add_constraints_from_ty(current, input, contra);
419 }
420 self.add_constraints_from_ty(current, sig.skip_binder().output(), variance);
421 }
422
423 fn add_constraints_from_region(
426 &mut self,
427 current: &CurrentItem,
428 region: ty::Region<'tcx>,
429 variance: VarianceTermPtr<'a>,
430 ) {
431 match *region {
432 ty::ReEarlyParam(ref data) => {
433 self.add_constraint(current, data.index, variance);
434 }
435
436 ty::ReStatic => {}
437
438 ty::ReBound(..) => {
439 }
444
445 ty::ReError(_) => {}
446
447 ty::ReLateParam(..) | ty::ReVar(..) | ty::RePlaceholder(..) | ty::ReErased => {
448 bug!(
451 "unexpected region encountered in variance \
452 inference: {:?}",
453 region
454 );
455 }
456 }
457 }
458
459 fn add_constraints_from_mt(
462 &mut self,
463 current: &CurrentItem,
464 mt: &ty::TypeAndMut<'tcx>,
465 variance: VarianceTermPtr<'a>,
466 ) {
467 match mt.mutbl {
468 hir::Mutability::Mut => {
469 let invar = self.invariant(variance);
470 self.add_constraints_from_ty(current, mt.ty, invar);
471 }
472
473 hir::Mutability::Not => {
474 self.add_constraints_from_ty(current, mt.ty, variance);
475 }
476 }
477 }
478}