Slightly refactor 'exiting_out loop in prepare_vtable_segments
1. Hide the option as an iterator, so it's nicer to work with 2. Replace a loop with `find`
This commit is contained in:
parent
364fc444a5
commit
d567f0fc68
@ -87,7 +87,7 @@ fn prepare_vtable_segments_inner<'tcx, T>(
|
|||||||
let mut visited = PredicateSet::new(tcx);
|
let mut visited = PredicateSet::new(tcx);
|
||||||
let predicate = trait_ref.without_const().to_predicate(tcx);
|
let predicate = trait_ref.without_const().to_predicate(tcx);
|
||||||
let mut stack: SmallVec<[(ty::PolyTraitRef<'tcx>, _, _); 5]> =
|
let mut stack: SmallVec<[(ty::PolyTraitRef<'tcx>, _, _); 5]> =
|
||||||
smallvec![(trait_ref, emit_vptr_on_new_entry, None)];
|
smallvec![(trait_ref, emit_vptr_on_new_entry, maybe_iter(None))];
|
||||||
visited.insert(predicate);
|
visited.insert(predicate);
|
||||||
|
|
||||||
// the main traversal loop:
|
// the main traversal loop:
|
||||||
@ -138,7 +138,7 @@ fn prepare_vtable_segments_inner<'tcx, T>(
|
|||||||
stack.push((
|
stack.push((
|
||||||
next_super_trait,
|
next_super_trait,
|
||||||
emit_vptr_on_new_entry,
|
emit_vptr_on_new_entry,
|
||||||
Some(direct_super_traits_iter),
|
maybe_iter(Some(direct_super_traits_iter)),
|
||||||
))
|
))
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -152,30 +152,26 @@ fn prepare_vtable_segments_inner<'tcx, T>(
|
|||||||
|
|
||||||
// emit innermost item, move to next sibling and stop there if possible, otherwise jump to outer level.
|
// emit innermost item, move to next sibling and stop there if possible, otherwise jump to outer level.
|
||||||
'exiting_out: loop {
|
'exiting_out: loop {
|
||||||
if let Some((inner_most_trait_ref, emit_vptr, siblings_opt)) = stack.last_mut() {
|
if let Some((inner_most_trait_ref, emit_vptr, siblings)) = stack.last_mut() {
|
||||||
segment_visitor(VtblSegment::TraitOwnEntries {
|
segment_visitor(VtblSegment::TraitOwnEntries {
|
||||||
trait_ref: *inner_most_trait_ref,
|
trait_ref: *inner_most_trait_ref,
|
||||||
emit_vptr: *emit_vptr,
|
emit_vptr: *emit_vptr,
|
||||||
})?;
|
})?;
|
||||||
|
|
||||||
'exiting_out_skip_visited_traits: loop {
|
match siblings.find(|&sibling| visited.insert(sibling.to_predicate(tcx))) {
|
||||||
if let Some(siblings) = siblings_opt {
|
Some(next_inner_most_trait_ref) => {
|
||||||
if let Some(next_inner_most_trait_ref) = siblings.next() {
|
// We're throwing away potential constness of super traits here.
|
||||||
if visited.insert(next_inner_most_trait_ref.to_predicate(tcx)) {
|
// FIXME: handle ~const super traits
|
||||||
// We're throwing away potential constness of super traits here.
|
let next_inner_most_trait_ref =
|
||||||
// FIXME: handle ~const super traits
|
next_inner_most_trait_ref.map_bound(|t| t.trait_ref);
|
||||||
let next_inner_most_trait_ref =
|
*inner_most_trait_ref = next_inner_most_trait_ref;
|
||||||
next_inner_most_trait_ref.map_bound(|t| t.trait_ref);
|
*emit_vptr = emit_vptr_on_new_entry;
|
||||||
*inner_most_trait_ref = next_inner_most_trait_ref;
|
break 'exiting_out;
|
||||||
*emit_vptr = emit_vptr_on_new_entry;
|
}
|
||||||
break 'exiting_out;
|
None => {
|
||||||
} else {
|
stack.pop();
|
||||||
continue 'exiting_out_skip_visited_traits;
|
continue 'exiting_out;
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
stack.pop();
|
|
||||||
continue 'exiting_out;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
// all done
|
// all done
|
||||||
@ -184,6 +180,12 @@ fn prepare_vtable_segments_inner<'tcx, T>(
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Turns option of iterator into an iterator (this is just flatten)
|
||||||
|
fn maybe_iter<I: Iterator>(i: Option<I>) -> impl Iterator<Item = I::Item> {
|
||||||
|
// Flatten is bad perf-vise, we could probably implement a special case here that is better
|
||||||
|
i.into_iter().flatten()
|
||||||
|
}
|
||||||
|
|
||||||
fn dump_vtable_entries<'tcx>(
|
fn dump_vtable_entries<'tcx>(
|
||||||
tcx: TyCtxt<'tcx>,
|
tcx: TyCtxt<'tcx>,
|
||||||
sp: Span,
|
sp: Span,
|
||||||
|
Loading…
x
Reference in New Issue
Block a user