Skip to content

Commit 5acc1ca

Browse files
committed
fix(resolver): avoid cloning when iterating using RcVecIter
1 parent 430fabe commit 5acc1ca

File tree

2 files changed

+32
-45
lines changed

2 files changed

+32
-45
lines changed

src/cargo/core/resolver/mod.rs

+10-10
Original file line numberDiff line numberDiff line change
@@ -446,13 +446,13 @@ fn activate_deps_loop(
446446
// conflict with us.
447447
let mut has_past_conflicting_dep = just_here_for_the_error_messages;
448448
if !has_past_conflicting_dep {
449-
if let Some(conflicting) = frame
450-
.remaining_siblings
451-
.clone()
452-
.filter_map(|(ref new_dep, _, _)| {
453-
past_conflicting_activations.conflicting(&resolver_ctx, new_dep)
454-
})
455-
.next()
449+
if let Some(conflicting) =
450+
frame
451+
.remaining_siblings
452+
.remaining()
453+
.find_map(|(ref new_dep, _, _)| {
454+
past_conflicting_activations.conflicting(&resolver_ctx, new_dep)
455+
})
456456
{
457457
// If one of our deps is known unresolvable
458458
// then we will not succeed.
@@ -757,7 +757,7 @@ impl RemainingCandidates {
757757
conflicting_prev_active: &mut ConflictMap,
758758
cx: &ResolverContext,
759759
) -> Option<(Summary, bool)> {
760-
for b in self.remaining.by_ref() {
760+
for b in self.remaining.iter() {
761761
let b_id = b.package_id();
762762
// The `links` key in the manifest dictates that there's only one
763763
// package in a dependency graph, globally, with that particular
@@ -783,7 +783,7 @@ impl RemainingCandidates {
783783
// Here we throw out our candidate if it's *compatible*, yet not
784784
// equal, to all previously activated versions.
785785
if let Some((a, _)) = cx.activations.get(&b_id.as_activations_key()) {
786-
if *a != b {
786+
if a != b {
787787
conflicting_prev_active
788788
.entry(a.package_id())
789789
.or_insert(ConflictReason::Semver);
@@ -796,7 +796,7 @@ impl RemainingCandidates {
796796
// necessarily return the item just yet. Instead we stash it away to
797797
// get returned later, and if we replaced something then that was
798798
// actually the candidate to try first so we return that.
799-
if let Some(r) = mem::replace(&mut self.has_another, Some(b)) {
799+
if let Some(r) = mem::replace(&mut self.has_another, Some(b.clone())) {
800800
return Some((r, true));
801801
}
802802
}

src/cargo/core/resolver/types.rs

+22-35
Original file line numberDiff line numberDiff line change
@@ -5,7 +5,6 @@ use crate::util::interning::InternedString;
55
use crate::util::GlobalContext;
66
use std::cmp::Ordering;
77
use std::collections::{BTreeMap, BTreeSet};
8-
use std::ops::Range;
98
use std::rc::Rc;
109
use std::time::{Duration, Instant};
1110

@@ -181,13 +180,13 @@ impl DepsFrame {
181180
fn min_candidates(&self) -> usize {
182181
self.remaining_siblings
183182
.peek()
184-
.map(|(_, (_, candidates, _))| candidates.len())
183+
.map(|(_, candidates, _)| candidates.len())
185184
.unwrap_or(0)
186185
}
187186

188-
pub fn flatten(&self) -> impl Iterator<Item = (PackageId, Dependency)> + '_ {
187+
pub fn flatten(&self) -> impl Iterator<Item = (PackageId, &Dependency)> + '_ {
189188
self.remaining_siblings
190-
.clone()
189+
.remaining()
191190
.map(move |(d, _, _)| (self.parent.package_id(), d))
192191
}
193192
}
@@ -251,15 +250,16 @@ impl RemainingDeps {
251250
// Figure out what our next dependency to activate is, and if nothing is
252251
// listed then we're entirely done with this frame (yay!) and we can
253252
// move on to the next frame.
254-
if let Some(sibling) = deps_frame.remaining_siblings.next() {
253+
let sibling = deps_frame.remaining_siblings.iter().next().cloned();
254+
if let Some(sibling) = sibling {
255255
let parent = Summary::clone(&deps_frame.parent);
256256
self.data.insert((deps_frame, insertion_time));
257257
return Some((just_here_for_the_error_messages, (parent, sibling)));
258258
}
259259
}
260260
None
261261
}
262-
pub fn iter(&mut self) -> impl Iterator<Item = (PackageId, Dependency)> + '_ {
262+
pub fn iter(&mut self) -> impl Iterator<Item = (PackageId, &Dependency)> + '_ {
263263
self.data.iter().flat_map(|(other, _)| other.flatten())
264264
}
265265
}
@@ -324,22 +324,27 @@ pub type ConflictMap = BTreeMap<PackageId, ConflictReason>;
324324

325325
pub struct RcVecIter<T> {
326326
vec: Rc<Vec<T>>,
327-
rest: Range<usize>,
327+
offset: usize,
328328
}
329329

330330
impl<T> RcVecIter<T> {
331331
pub fn new(vec: Rc<Vec<T>>) -> RcVecIter<T> {
332-
RcVecIter {
333-
rest: 0..vec.len(),
334-
vec,
335-
}
332+
RcVecIter { vec, offset: 0 }
333+
}
334+
335+
pub fn peek(&self) -> Option<&T> {
336+
self.vec.get(self.offset)
336337
}
337338

338-
fn peek(&self) -> Option<(usize, &T)> {
339-
self.rest
340-
.clone()
341-
.next()
342-
.and_then(|i| self.vec.get(i).map(|val| (i, &*val)))
339+
pub fn remaining(&self) -> impl Iterator<Item = &T> + '_ {
340+
self.vec.get(self.offset..).into_iter().flatten()
341+
}
342+
343+
pub fn iter(&mut self) -> impl Iterator<Item = &T> + '_ {
344+
let iter = self.vec.get(self.offset..).into_iter().flatten();
345+
// This call to `ìnspect()` is used to increment `self.offset` when iterating the inner `Vec`,
346+
// while keeping the `ExactSizeIterator` property.
347+
iter.inspect(|_| self.offset += 1)
343348
}
344349
}
345350

@@ -348,25 +353,7 @@ impl<T> Clone for RcVecIter<T> {
348353
fn clone(&self) -> RcVecIter<T> {
349354
RcVecIter {
350355
vec: self.vec.clone(),
351-
rest: self.rest.clone(),
356+
offset: self.offset,
352357
}
353358
}
354359
}
355-
356-
impl<T> Iterator for RcVecIter<T>
357-
where
358-
T: Clone,
359-
{
360-
type Item = T;
361-
362-
fn next(&mut self) -> Option<Self::Item> {
363-
self.rest.next().and_then(|i| self.vec.get(i).cloned())
364-
}
365-
366-
fn size_hint(&self) -> (usize, Option<usize>) {
367-
// rest is a std::ops::Range, which is an ExactSizeIterator.
368-
self.rest.size_hint()
369-
}
370-
}
371-
372-
impl<T: Clone> ExactSizeIterator for RcVecIter<T> {}

0 commit comments

Comments
 (0)