Struct rustc_trait_selection::solve::search_graph::SearchGraph
source · pub(super) struct SearchGraph<'tcx> {
mode: SolverMode,
stack: IndexVec<StackDepth, StackEntry<'tcx>>,
provisional_cache: FxHashMap<CanonicalInput<'tcx>, ProvisionalCacheEntry<'tcx>>,
cycle_participants: FxHashSet<CanonicalInput<'tcx>>,
}
Fields§
§mode: SolverMode
§stack: IndexVec<StackDepth, StackEntry<'tcx>>
The stack of goals currently being computed.
An element is deeper in the stack if its index is lower.
provisional_cache: FxHashMap<CanonicalInput<'tcx>, ProvisionalCacheEntry<'tcx>>
§cycle_participants: FxHashSet<CanonicalInput<'tcx>>
We put only the root goal of a coinductive cycle into the global cache.
If we were to use that result when later trying to prove another cycle participant, we can end up with unstable query results.
See tests/ui/next-solver/coinduction/incompleteness-unstable-result.rs for an example of where this is needed.
Implementations§
source§impl<'tcx> SearchGraph<'tcx>
impl<'tcx> SearchGraph<'tcx>
pub(super) fn new(mode: SolverMode) -> SearchGraph<'tcx>
pub(super) fn solver_mode(&self) -> SolverMode
sourcefn on_cache_hit(&mut self, additional_depth: usize, encountered_overflow: bool)
fn on_cache_hit(&mut self, additional_depth: usize, encountered_overflow: bool)
Update the stack and reached depths on cache hits.
sourcefn pop_stack(&mut self) -> StackEntry<'tcx>
fn pop_stack(&mut self) -> StackEntry<'tcx>
Pops the highest goal from the stack, lazily updating the the next goal in the stack.
Directly popping from the stack instead of using this method would cause us to not track overflow and recursion depth correctly.
sourcepub(super) fn global_cache(
&self,
tcx: TyCtxt<'tcx>
) -> &'tcx EvaluationCache<'tcx>
pub(super) fn global_cache( &self, tcx: TyCtxt<'tcx> ) -> &'tcx EvaluationCache<'tcx>
The trait solver behavior is different for coherence so we use a separate cache. Alternatively we could use a single cache and share it between coherence and ordinary trait solving.
pub(super) fn is_empty(&self) -> bool
pub(super) fn current_goal_is_normalizes_to(&self) -> bool
sourcefn allowed_depth_for_nested(
tcx: TyCtxt<'tcx>,
stack: &IndexVec<StackDepth, StackEntry<'tcx>>
) -> Option<Limit>
fn allowed_depth_for_nested( tcx: TyCtxt<'tcx>, stack: &IndexVec<StackDepth, StackEntry<'tcx>> ) -> Option<Limit>
Returns the remaining depth allowed for nested goals.
This is generally simply one less than the current depth. However, if we encountered overflow, we significantly reduce the remaining depth of all nested goals to prevent hangs in case there is exponential blowup.
fn stack_coinductive_from( tcx: TyCtxt<'tcx>, stack: &IndexVec<StackDepth, StackEntry<'tcx>>, head: StackDepth ) -> bool
fn tag_cycle_participants( stack: &mut IndexVec<StackDepth, StackEntry<'tcx>>, cycle_participants: &mut FxHashSet<CanonicalInput<'tcx>>, usage_kind: HasBeenUsed, head: StackDepth )
fn clear_dependent_provisional_results( provisional_cache: &mut FxHashMap<CanonicalInput<'tcx>, ProvisionalCacheEntry<'tcx>>, head: StackDepth )
sourcepub(super) fn with_new_goal(
&mut self,
tcx: TyCtxt<'tcx>,
input: CanonicalInput<'tcx>,
inspect: &mut ProofTreeBuilder<'tcx>,
prove_goal: impl FnMut(&mut Self, &mut ProofTreeBuilder<'tcx>) -> QueryResult<'tcx>
) -> QueryResult<'tcx>
pub(super) fn with_new_goal( &mut self, tcx: TyCtxt<'tcx>, input: CanonicalInput<'tcx>, inspect: &mut ProofTreeBuilder<'tcx>, prove_goal: impl FnMut(&mut Self, &mut ProofTreeBuilder<'tcx>) -> QueryResult<'tcx> ) -> QueryResult<'tcx>
Probably the most involved method of the whole solver.
Given some goal which is proven via the prove_goal
closure, this
handles caching, overflow, and coinductive cycles.
fn response_no_constraints( tcx: TyCtxt<'tcx>, goal: CanonicalInput<'tcx>, certainty: Certainty ) -> QueryResult<'tcx>
Auto Trait Implementations§
impl<'tcx> DynSend for SearchGraph<'tcx>
impl<'tcx> DynSync for SearchGraph<'tcx>
impl<'tcx> Freeze for SearchGraph<'tcx>
impl<'tcx> !RefUnwindSafe for SearchGraph<'tcx>
impl<'tcx> Send for SearchGraph<'tcx>
impl<'tcx> Sync for SearchGraph<'tcx>
impl<'tcx> Unpin for SearchGraph<'tcx>
impl<'tcx> !UnwindSafe for SearchGraph<'tcx>
Blanket Implementations§
source§impl<T> BorrowMut<T> for Twhere
T: ?Sized,
impl<T> BorrowMut<T> for Twhere
T: ?Sized,
source§fn borrow_mut(&mut self) -> &mut T
fn borrow_mut(&mut self) -> &mut T
source§impl<T, R> CollectAndApply<T, R> for T
impl<T, R> CollectAndApply<T, R> for T
§impl<T> Filterable for T
impl<T> Filterable for T
source§impl<T> Instrument for T
impl<T> Instrument for T
source§fn instrument(self, span: Span) -> Instrumented<Self>
fn instrument(self, span: Span) -> Instrumented<Self>
source§fn in_current_span(self) -> Instrumented<Self>
fn in_current_span(self) -> Instrumented<Self>
source§impl<P> IntoQueryParam<P> for P
impl<P> IntoQueryParam<P> for P
fn into_query_param(self) -> P
source§impl<T> MaybeResult<T> for T
impl<T> MaybeResult<T> for T
§impl<T> Pointable for T
impl<T> Pointable for T
source§impl<'tcx, T> ToPredicate<'tcx, T> for T
impl<'tcx, T> ToPredicate<'tcx, T> for T
fn to_predicate(self, _tcx: TyCtxt<'tcx>) -> T
source§impl<Tcx, T> Value<Tcx> for Twhere
Tcx: DepContext,
impl<Tcx, T> Value<Tcx> for Twhere
Tcx: DepContext,
default fn from_cycle_error( tcx: Tcx, cycle_error: &CycleError, _guar: ErrorGuaranteed ) -> T
source§impl<T> WithSubscriber for T
impl<T> WithSubscriber for T
source§fn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>
fn with_subscriber<S>(self, subscriber: S) -> WithDispatch<Self>
source§fn with_current_subscriber(self) -> WithDispatch<Self>
fn with_current_subscriber(self) -> WithDispatch<Self>
impl<'a, T> Captures<'a> for Twhere
T: ?Sized,
impl<T> ErasedDestructor for Twhere
T: 'static,
impl<T> MaybeSendSync for T
Layout§
Note: Most layout information is completely unstable and may even differ between compilations. The only exception is types with certain repr(...)
attributes. Please see the Rust Reference's “Type Layout” chapter for details on type layout guarantees.
Size: 96 bytes