|
| 1 | +# Invariants of the type system |
| 2 | + |
| 3 | +FIXME: This file talks about invariants of the type system as a whole, not only the solver |
| 4 | + |
| 5 | +There are a lot of invariants - things the type system guarantees to be true at all times - |
| 6 | +which are desirable or expected from other languages and type systems. Unfortunately, quite |
| 7 | +a few of them do not hold in Rust right now. This is either a fundamental to its design or |
| 8 | +caused by bugs and something that may change in the future. |
| 9 | + |
| 10 | +It is important to know about the things you can assume while working on - and with - the |
| 11 | +type system, so here's an incomplete and inofficial list of invariants of |
| 12 | +the core type system: |
| 13 | + |
| 14 | +- ✅: this invariant mostly holds, with some weird exceptions, you can rely on it outside |
| 15 | +of these cases |
| 16 | +- ❌: this invariant does not hold, either due to bugs or by design, you must not rely on |
| 17 | +it for soundness or have to be incredibly careful when doing so |
| 18 | + |
| 19 | +### `wf(X)` implies `wf(normalize(X))` ✅ |
| 20 | + |
| 21 | +If a type containing aliases is well-formed, it should also be |
| 22 | +well-formed after normalizing said aliases. We rely on this as |
| 23 | +otherwise we would have to re-check for well-formedness for these |
| 24 | +types. |
| 25 | + |
| 26 | +This is unfortunately broken for `<fndef as FnOnce<..>>::Output` due to implied bounds, |
| 27 | +resulting in [#114936]. |
| 28 | + |
| 29 | +### applying inference results from a goal does not change its result ❌ |
| 30 | + |
| 31 | +TODO: this invariant is formulated in a weird way and needs to be elaborated. |
| 32 | +Pretty much: I would like this check to only fail if there's a solver bug: |
| 33 | +https://github.com/rust-lang/rust/blob/2ffeb4636b4ae376f716dc4378a7efb37632dc2d/compiler/rustc_trait_selection/src/solve/eval_ctxt.rs#L391-L407 |
| 34 | + |
| 35 | +If we prove some goal/equate types/whatever, apply the resulting inference constraints, |
| 36 | +and then redo the original action, the result should be the same. |
| 37 | + |
| 38 | +This unfortunately does not hold - at least in the new solver - due to a few annoying reasons. |
| 39 | + |
| 40 | +### The trait solver has to be *locally sound* ✅ |
| 41 | + |
| 42 | +This means that we must never return *success* for goals for which no `impl` exists. That would |
| 43 | +mean we assume a trait is implemented even though it is not, which is very likely to result in |
| 44 | +actual unsoundness. When using `where`-bounds to prove a goal, the `impl` will be provided by the |
| 45 | +user of the item. |
| 46 | + |
| 47 | +This invariant only holds if we check region constraints. As we do not check region constraints |
| 48 | +during implicit negative overlap check in coherence, this invariant is broken there. As this check |
| 49 | +relies on *completeness* of the trait solver, it is not able to use the current region constraints |
| 50 | +check - `InferCtxt::resolve_regions` - as its handling of type outlives goals is incomplete. |
| 51 | + |
| 52 | +### Normalization of semantically equal aliases in empty environments results in a unique type ✅ |
| 53 | + |
| 54 | +Normalization for alias types/consts has to have a unique result. Otherwise we can easily |
| 55 | +implement transmute in safe code. Given the following function, we have to make sure that |
| 56 | +the input and output types always get normalized to the same concrete type. |
| 57 | + |
| 58 | +```rust |
| 59 | +fn foo<T: Trait>( |
| 60 | + x: <T as Trait>::Assoc |
| 61 | +) -> <T as Trait>::Assoc { |
| 62 | + x |
| 63 | +} |
| 64 | +``` |
| 65 | + |
| 66 | +Many of the currently known unsound issues end up relying on this invariant being broken. |
| 67 | +It is however very difficult to imagine a sound type system without this invariant, so |
| 68 | +the issue is that the invariant is broken, not that we incorrectly rely on it. |
| 69 | + |
| 70 | +### Generic goals and their instantiations have the same result ✅ |
| 71 | + |
| 72 | +Pretty much: If we successfully typecheck a generic function concrete instantiations |
| 73 | +of that function should also typeck. We should not get errors post-monomorphization. |
| 74 | +We can however get overflow errors at that point. |
| 75 | + |
| 76 | +TODO: example for overflow error post-monomorphization |
| 77 | + |
| 78 | +This invariant is relied on to allow the normalization of generic aliases. Breaking |
| 79 | +it can easily result in unsoundness, e.g. [#57893](https://github.com/rust-lang/rust/issues/57893) |
| 80 | + |
| 81 | +### Trait goals in empty environments are proven by a unique impl ✅ |
| 82 | + |
| 83 | +If a trait goal holds with an empty environment, there should be a unique `impl`, |
| 84 | +either user-defined or builtin, which is used to prove that goal. This is |
| 85 | +necessary to select a unique method. It |
| 86 | + |
| 87 | +We do however break this invariant in few cases, some of which are due to bugs, |
| 88 | +some by design: |
| 89 | +- *marker traits* are allowed to overlap as they do not have associated items |
| 90 | +- *specialization* allows specializing impls to overlap with their parent |
| 91 | +- the builtin trait object trait implementation can overlap with a user-defined impl: |
| 92 | +[#57893] |
| 93 | + |
| 94 | +### The type system is complete ❌ |
| 95 | + |
| 96 | +The type system is not complete, it often adds unnecessary inference constraints, and errors |
| 97 | +even though the goal could hold. |
| 98 | + |
| 99 | +- method selection |
| 100 | +- opaque type inference |
| 101 | +- handling type outlives constraints |
| 102 | +- preferring `ParamEnv` candidates over `Impl` candidates during candidate selection |
| 103 | +in the trait solver |
| 104 | + |
| 105 | +#### The type system is complete during the implicit negative overlap check in coherence ✅ |
| 106 | + |
| 107 | +During the implicit negative overlap check in coherence we must never return *error* for |
| 108 | +goals which can be proven. This would allow for overlapping impls with potentially different |
| 109 | +associated items, breaking a bunch of other invariants. |
| 110 | + |
| 111 | +This invariant is currently broken in many different ways while actually something we rely on. |
| 112 | +We have to be careful as it is quite easy to break: |
| 113 | +- generalization of aliases |
| 114 | +- generalization during subtyping binders (luckily not exploitable in coherence) |
| 115 | + |
| 116 | +### Trait solving must be (free) lifetime agnostic ✅ |
| 117 | + |
| 118 | +Trait solving during codegen should have the same result as during typeck. As we erase |
| 119 | +all free regions during codegen we must not rely on them during typeck. A noteworthy example |
| 120 | +is special behavior for `'static`. |
| 121 | + |
| 122 | +We also have to be careful with relying on equality of regions in the trait solver. |
| 123 | +This is fine for codegen, as we treat all erased regions as equal. We can however |
| 124 | +lose equality information from HIR to MIR typeck. |
| 125 | + |
| 126 | +The new solver "uniquifies regions" during canonicalization, canonicalizing `u32: Trait<'x, 'x>` |
| 127 | +as `exists<'0, '1> u32: Trait<'0, '1>`, to make it harder to rely on this property. |
| 128 | + |
| 129 | +### Removing ambiguity makes strictly more things compile ❌ |
| 130 | + |
| 131 | +Ideally we *should* not rely on ambiguity for things to compile. |
| 132 | +Not doing that will cause future improvements to be breaking changes. |
| 133 | + |
| 134 | +Due to *incompleteness* this is not the case and improving inference can result in inference |
| 135 | +changes, breaking existing projects. |
| 136 | + |
| 137 | +### Semantic equality implies structural equality ✅ |
| 138 | + |
| 139 | +Two types being equal in the type system must mean that they have the |
| 140 | +same `TypeId` after instantiating their generic parameters with concrete |
| 141 | +arguments. This currently does not hold: [#97156]. |
| 142 | + |
| 143 | +[#57893]: https://github.com/rust-lang/rust/issues/57893 |
| 144 | +[#97156]: https://github.com/rust-lang/rust/issues/97156 |
| 145 | +[#114936]: https://github.com/rust-lang/rust/issues/114936 |
0 commit comments