File: provisional-cache-impacts-behavior.rs

package info (click to toggle)
rustc 1.85.0%2Bdfsg2-3
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid, trixie
  • size: 893,176 kB
  • sloc: xml: 158,127; python: 35,830; javascript: 19,497; cpp: 19,002; sh: 17,245; ansic: 13,127; asm: 4,376; makefile: 1,051; lisp: 29; perl: 29; ruby: 19; sql: 11
file content (70 lines) | stat: -rw-r--r-- 1,858 bytes parent folder | download | duplicates (6)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
//@ compile-flags: -Znext-solver
//@ check-pass
#![feature(rustc_attrs)]

// A test showcasing that using a provisional cache can differ
// from only tracking stack entries.
//
// Without a provisional cache, we have the following proof tree:
//
// - (): A
//   - (): B
//     - (): A (coinductive cycle)
//     - (): C
//       - (): B (coinductive cycle)
//   - (): C
//     - (): B
//        - (): A (coinductive cycle)
//        - (): C (coinductive cycle)
//
// While with the current provisional cache implementation we get:
//
// - (): A
//   - (): B
//     - (): A (coinductive cycle)
//     - (): C
//       - (): B (coinductive cycle)
//   - (): C
//     - (): B (provisional cache hit)
//
// Note that if even if we were to expand the provisional cache hit,
// the proof tree would still be different:
//
// - (): A
//   - (): B
//     - (): A (coinductive cycle)
//     - (): C
//       - (): B (coinductive cycle)
//   - (): C
//     - (): B (provisional cache hit, expanded)
//       - (): A (coinductive cycle)
//       - (): C
//         - (): B (coinductive cycle)
//
// Theoretically, this can result in observable behavior differences
// due to incompleteness. However, this would require a very convoluted
// example and would still be sound. The difference is determinstic
// and can not be observed outside of the cycle itself as we don't move
// non-root cycle participants into the global cache.
//
// For an example of how incompleteness could impact the observable behavior here, see
//
//   tests/ui/traits/next-solver/cycles/coinduction/incompleteness-unstable-result.rs
#[rustc_coinductive]
trait A {}

#[rustc_coinductive]
trait B {}

#[rustc_coinductive]
trait C {}

impl<T: ?Sized + B + C> A for T {}
impl<T: ?Sized + A + C> B for T {}
impl<T: ?Sized + B> C for T {}

fn impls_a<T: A>() {}

fn main() {
    impls_a::<()>();
}