Skip to content

Commit 4ffe98a

Browse files
committed
store jobs priorities in the pending queue
This cleans up the priority-sorted scheduling by removing the need for a priority accessor that would hash the nodes, and allows inserting in the queue at the correctly sorted position to remove the insert + sort combination.
1 parent 9ef2958 commit 4ffe98a

File tree

2 files changed

+29
-33
lines changed

2 files changed

+29
-33
lines changed

src/cargo/core/compiler/job_queue.rs

+15-12
Original file line numberDiff line numberDiff line change
@@ -162,7 +162,7 @@ struct DrainState<'cfg> {
162162
/// The list of jobs that we have not yet started executing, but have
163163
/// retrieved from the `queue`. We eagerly pull jobs off the main queue to
164164
/// allow us to request jobserver tokens pretty early.
165-
pending_queue: Vec<(Unit, Job)>,
165+
pending_queue: Vec<(Unit, Job, usize)>,
166166
print: DiagnosticPrinter<'cfg>,
167167

168168
/// How many jobs we've finished
@@ -576,26 +576,29 @@ impl<'cfg> DrainState<'cfg> {
576576
// possible that can run. Note that this is also the point where we
577577
// start requesting job tokens. Each job after the first needs to
578578
// request a token.
579-
while let Some((unit, job)) = self.queue.dequeue() {
580-
self.pending_queue.push((unit, job));
579+
while let Some((unit, job, priority)) = self.queue.dequeue() {
580+
// We want to keep the pieces of work in the `pending_queue` sorted
581+
// by their priorities, and insert the current job at its correctly
582+
// sorted position: following the lower priority jobs, and the ones
583+
// with the same priority (since they were dequeued before the
584+
// current one, we also keep that relation).
585+
let idx = self
586+
.pending_queue
587+
.partition_point(|&(_, _, p)| p <= priority);
588+
self.pending_queue.insert(idx, (unit, job, priority));
581589
if self.active.len() + self.pending_queue.len() > 1 {
582590
jobserver_helper.request_token();
583591
}
584592
}
585593

586-
// If multiple pieces of work are waiting in the pending queue, we can
587-
// sort it according to their priorities: higher priorities should be
588-
// scheduled sooner.
589-
self.pending_queue
590-
.sort_by_cached_key(|(unit, _)| self.queue.priority(unit));
591-
592594
// Now that we've learned of all possible work that we can execute
593595
// try to spawn it so long as we've got a jobserver token which says
594596
// we're able to perform some parallel work.
595-
// The `pending_queue` is sorted in ascending priority order, and we're
596-
// removing the highest priority items from its end.
597+
// The `pending_queue` is sorted in ascending priority order, and we
598+
// remove items from its end to schedule the highest priority items
599+
// sooner.
597600
while self.has_extra_tokens() && !self.pending_queue.is_empty() {
598-
let (unit, job) = self.pending_queue.pop().unwrap();
601+
let (unit, job, _) = self.pending_queue.pop().unwrap();
599602
*self.counts.get_mut(&unit.pkg.package_id()).unwrap() -= 1;
600603
if !cx.bcx.build_config.build_plan {
601604
// Print out some nice progress information.

src/cargo/util/dependency_queue.rs

+14-21
Original file line numberDiff line numberDiff line change
@@ -149,15 +149,15 @@ impl<N: Hash + Eq + Clone, E: Eq + Hash + Clone, V> DependencyQueue<N, E, V> {
149149
///
150150
/// A package is ready to be built when it has 0 un-built dependencies. If
151151
/// `None` is returned then no packages are ready to be built.
152-
pub fn dequeue(&mut self) -> Option<(N, V)> {
153-
let key = self
152+
pub fn dequeue(&mut self) -> Option<(N, V, usize)> {
153+
let (key, priority) = self
154154
.dep_map
155155
.iter()
156156
.filter(|(_, (deps, _))| deps.is_empty())
157-
.map(|(key, _)| key.clone())
158-
.max_by_key(|k| self.priority[k])?;
157+
.map(|(key, _)| (key.clone(), self.priority[key]))
158+
.max_by_key(|(_, priority)| *priority)?;
159159
let (_, data) = self.dep_map.remove(&key).unwrap();
160-
Some((key, data))
160+
Some((key, data, priority))
161161
}
162162

163163
/// Returns `true` if there are remaining packages to be built.
@@ -170,13 +170,6 @@ impl<N: Hash + Eq + Clone, E: Eq + Hash + Clone, V> DependencyQueue<N, E, V> {
170170
self.dep_map.len()
171171
}
172172

173-
/// Returns the relative priority of a node. Higher priorities should be scheduled sooner.
174-
/// Currently computed as the transitive cost of the given node: its own, plus the cost of its
175-
/// reverse dependencies.
176-
pub(crate) fn priority(&self, node: &N) -> usize {
177-
self.priority[node]
178-
}
179-
180173
/// Indicate that something has finished.
181174
///
182175
/// Calling this function indicates that the `node` has produced `edge`. All
@@ -220,19 +213,19 @@ mod test {
220213
q.queue(5, (), vec![(4, ()), (3, ())], 1);
221214
q.queue_finished();
222215

223-
assert_eq!(q.dequeue(), Some((1, ())));
224-
assert_eq!(q.dequeue(), Some((3, ())));
216+
assert_eq!(q.dequeue(), Some((1, (), 5)));
217+
assert_eq!(q.dequeue(), Some((3, (), 4)));
225218
assert_eq!(q.dequeue(), None);
226219
q.finish(&3, &());
227220
assert_eq!(q.dequeue(), None);
228221
q.finish(&1, &());
229-
assert_eq!(q.dequeue(), Some((2, ())));
222+
assert_eq!(q.dequeue(), Some((2, (), 4)));
230223
assert_eq!(q.dequeue(), None);
231224
q.finish(&2, &());
232-
assert_eq!(q.dequeue(), Some((4, ())));
225+
assert_eq!(q.dequeue(), Some((4, (), 3)));
233226
assert_eq!(q.dequeue(), None);
234227
q.finish(&4, &());
235-
assert_eq!(q.dequeue(), Some((5, ())));
228+
assert_eq!(q.dequeue(), Some((5, (), 2)));
236229
}
237230

238231
#[test]
@@ -245,16 +238,16 @@ mod test {
245238
q.queue(4, (), vec![(2, ()), (3, ())], 1);
246239
q.queue_finished();
247240

248-
assert_eq!(q.dequeue(), Some((3, ())));
249-
assert_eq!(q.dequeue(), Some((1, ())));
241+
assert_eq!(q.dequeue(), Some((3, (), 9)));
242+
assert_eq!(q.dequeue(), Some((1, (), 4)));
250243
assert_eq!(q.dequeue(), None);
251244
q.finish(&3, &());
252245
assert_eq!(q.dequeue(), None);
253246
q.finish(&1, &());
254-
assert_eq!(q.dequeue(), Some((2, ())));
247+
assert_eq!(q.dequeue(), Some((2, (), 3)));
255248
assert_eq!(q.dequeue(), None);
256249
q.finish(&2, &());
257-
assert_eq!(q.dequeue(), Some((4, ())));
250+
assert_eq!(q.dequeue(), Some((4, (), 2)));
258251
assert_eq!(q.dequeue(), None);
259252
q.finish(&4, &());
260253
assert_eq!(q.dequeue(), None);

0 commit comments

Comments
 (0)