rustc_query_system/dep_graph/
mod.rs1pub mod debug;
2pub mod dep_node;
3mod edges;
4mod graph;
5mod query;
6mod serialized;
7
8use std::panic;
9
10pub use dep_node::{DepKind, DepKindStruct, DepNode, DepNodeParams, WorkProductId};
11pub(crate) use graph::DepGraphData;
12pub use graph::{DepGraph, DepNodeIndex, TaskDepsRef, WorkProduct, WorkProductMap, hash_result};
13pub use query::DepGraphQuery;
14use rustc_data_structures::profiling::SelfProfilerRef;
15use rustc_data_structures::sync::DynSync;
16use rustc_session::Session;
17pub use serialized::{SerializedDepGraph, SerializedDepNodeIndex};
18use tracing::instrument;
19
20use self::graph::{MarkFrame, print_markframe_trace};
21use crate::ich::StableHashingContext;
22
23pub trait DepContext: Copy {
24    type Deps: Deps;
25
26    fn with_stable_hashing_context<R>(self, f: impl FnOnce(StableHashingContext<'_>) -> R) -> R;
28
29    fn dep_graph(&self) -> &DepGraph<Self::Deps>;
31
32    fn profiler(&self) -> &SelfProfilerRef;
34
35    fn sess(&self) -> &Session;
37
38    fn dep_kind_info(&self, dep_node: DepKind) -> &DepKindStruct<Self>;
39
40    #[inline(always)]
41    fn fingerprint_style(self, kind: DepKind) -> FingerprintStyle {
42        let data = self.dep_kind_info(kind);
43        if data.is_anon {
44            return FingerprintStyle::Opaque;
45        }
46        data.fingerprint_style
47    }
48
49    #[inline(always)]
50    fn is_eval_always(self, kind: DepKind) -> bool {
52        self.dep_kind_info(kind).is_eval_always
53    }
54
55    #[inline]
61    #[instrument(skip(self, frame), level = "debug")]
62    fn try_force_from_dep_node(
63        self,
64        dep_node: DepNode,
65        prev_index: SerializedDepNodeIndex,
66        frame: &MarkFrame<'_>,
67    ) -> bool {
68        let cb = self.dep_kind_info(dep_node.kind);
69        if let Some(f) = cb.force_from_dep_node {
70            match panic::catch_unwind(panic::AssertUnwindSafe(|| f(self, dep_node, prev_index))) {
71                Err(value) => {
72                    if !value.is::<rustc_errors::FatalErrorMarker>() {
73                        print_markframe_trace(self.dep_graph(), frame);
74                    }
75                    panic::resume_unwind(value)
76                }
77                Ok(query_has_been_forced) => query_has_been_forced,
78            }
79        } else {
80            false
81        }
82    }
83
84    fn try_load_from_on_disk_cache(self, dep_node: DepNode) {
86        let cb = self.dep_kind_info(dep_node.kind);
87        if let Some(f) = cb.try_load_from_on_disk_cache {
88            f(self, dep_node)
89        }
90    }
91
92    fn with_reduced_queries<T>(self, _: impl FnOnce() -> T) -> T;
93}
94
95pub trait Deps: DynSync {
96    fn with_deps<OP, R>(deps: TaskDepsRef<'_>, op: OP) -> R
98    where
99        OP: FnOnce() -> R;
100
101    fn read_deps<OP>(op: OP)
103    where
104        OP: for<'a> FnOnce(TaskDepsRef<'a>);
105
106    fn name(&self, dep_kind: DepKind) -> &'static str;
107
108    const DEP_KIND_NULL: DepKind;
110
111    const DEP_KIND_RED: DepKind;
113
114    const DEP_KIND_SIDE_EFFECT: DepKind;
116
117    const DEP_KIND_ANON_ZERO_DEPS: DepKind;
119
120    const DEP_KIND_MAX: u16;
123}
124
125pub trait HasDepContext: Copy {
126    type Deps: self::Deps;
127    type DepContext: self::DepContext<Deps = Self::Deps>;
128
129    fn dep_context(&self) -> &Self::DepContext;
130}
131
132impl<T: DepContext> HasDepContext for T {
133    type Deps = T::Deps;
134    type DepContext = Self;
135
136    fn dep_context(&self) -> &Self::DepContext {
137        self
138    }
139}
140
141impl<T: HasDepContext, Q: Copy> HasDepContext for (T, Q) {
142    type Deps = T::Deps;
143    type DepContext = T::DepContext;
144
145    fn dep_context(&self) -> &Self::DepContext {
146        self.0.dep_context()
147    }
148}
149
150#[derive(Debug, PartialEq, Eq, Copy, Clone)]
152pub enum FingerprintStyle {
153    DefPathHash,
155    HirId,
157    Unit,
159    Opaque,
161}
162
163impl FingerprintStyle {
164    #[inline]
165    pub const fn reconstructible(self) -> bool {
166        match self {
167            FingerprintStyle::DefPathHash | FingerprintStyle::Unit | FingerprintStyle::HirId => {
168                true
169            }
170            FingerprintStyle::Opaque => false,
171        }
172    }
173}