Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

MultiProgress: prune 'zombie' progress bars; fixes #426; add render tests #438

Merged
merged 3 commits into from
Jun 13, 2022
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
18 changes: 18 additions & 0 deletions src/draw_target.rs
Original file line number Diff line number Diff line change
Expand Up @@ -187,6 +187,10 @@ impl ProgressDrawTarget {
_ => None,
}
}

pub(crate) fn last_line_count(&mut self) -> Option<&mut usize> {
self.kind.last_line_count()
}
}

#[derive(Debug)]
Expand All @@ -209,6 +213,20 @@ enum TargetKind {
},
}

impl TargetKind {
fn last_line_count(&mut self) -> Option<&mut usize> {
match self {
TargetKind::Term {
last_line_count, ..
} => Some(last_line_count),
TargetKind::TermLike {
last_line_count, ..
} => Some(last_line_count),
_ => None,
}
}
}

pub(crate) enum Drawable<'a> {
Term {
term: &'a Term,
Expand Down
5 changes: 5 additions & 0 deletions src/in_memory.rs
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,11 @@ impl InMemoryTerm {
}
}

pub fn reset(&self) {
let mut state = self.state.lock().unwrap();
*state = InMemoryTermState::new(state.parser.screen().size().0, state.width);
}

pub fn contents(&self) -> String {
let state = self.state.lock().unwrap();

Expand Down
110 changes: 80 additions & 30 deletions src/multi.rs
Original file line number Diff line number Diff line change
@@ -1,9 +1,11 @@
use std::fmt::{Debug, Formatter};
use std::io;
use std::sync::{Arc, RwLock};
use std::sync::{Arc, Mutex, RwLock, Weak};
use std::time::Instant;

use crate::draw_target::{DrawState, DrawStateWrapper, ProgressDrawTarget};
use crate::progress_bar::ProgressBar;
use crate::state::BarState;

/// Manages multiple progress bars from different threads
#[derive(Debug, Clone)]
Expand Down Expand Up @@ -128,8 +130,11 @@ impl MultiProgress {
}

fn internalize(&self, location: InsertLocation, pb: ProgressBar) -> ProgressBar {
let idx = self.state.write().unwrap().insert(location);
let mut state = self.state.write().unwrap();

let idx = state.insert(location);
pb.set_draw_target(ProgressDrawTarget::new_remote(self.state.clone(), idx));
state.members.get_mut(idx).unwrap().pb = pb.weak_bar_state();
pb
}

Expand Down Expand Up @@ -168,9 +173,9 @@ impl MultiProgress {
#[derive(Debug)]
pub(crate) struct MultiState {
/// The collection of states corresponding to progress bars
/// the state is None for bars that have not yet been drawn or have been removed
draw_states: Vec<Option<DrawState>>,
/// Set of removed bars, should have corresponding `None` elements in the `draw_states` vector
members: Vec<MultiStateMember>,
/// Set of removed bars, should have corresponding members in the `members` vector with a
/// `draw_state` of `None`.
free_set: Vec<usize>,
/// Indices to the `draw_states` to maintain correct visual order
ordering: Vec<usize>,
Expand All @@ -187,7 +192,7 @@ pub(crate) struct MultiState {
impl MultiState {
fn new(draw_target: ProgressDrawTarget) -> Self {
Self {
draw_states: vec![],
members: vec![],
free_set: vec![],
ordering: vec![],
draw_target,
Expand All @@ -203,6 +208,32 @@ impl MultiState {
extra_lines: Option<Vec<String>>,
now: Instant,
) -> io::Result<()> {
// Reap all consecutive 'zombie' progress bars from head of the list
let mut zombies = vec![];
djc marked this conversation as resolved.
Show resolved Hide resolved
let mut adjust = 0;
for index in self.ordering.iter() {
let member = &self.members[*index];
if !member.is_zombie {
break;
}

zombies.push(*index);
adjust += member
.draw_state
.as_ref()
.map(|d| d.lines.len())
.unwrap_or_default();
}

for index in zombies {
self.remove_idx(index);
}

// Adjust last line count so we don't clear too many lines
if let Some(last_line_count) = self.draw_target.last_line_count() {
*last_line_count -= adjust;
}

let orphan_lines_count = self.orphan_lines.len();
force_draw |= orphan_lines_count > 0;
let mut drawable = match self.draw_target.drawable(force_draw, now) {
Expand All @@ -222,9 +253,15 @@ impl MultiState {
draw_state.lines.append(&mut self.orphan_lines);

for index in self.ordering.iter() {
if let Some(state) = &self.draw_states[*index] {
let member = &mut self.members[*index];
if let Some(state) = &member.draw_state {
draw_state.lines.extend_from_slice(&state.lines[..]);
}

// Mark the dead progress bar as a zombie - will be reaped on next draw
if member.pb.upgrade().is_none() {
member.is_zombie = true;
}
}

drop(draw_state);
Expand All @@ -244,20 +281,14 @@ impl MultiState {
}

pub(crate) fn draw_state(&mut self, idx: usize) -> DrawStateWrapper<'_> {
let (states, orphans) = (&mut self.draw_states, &mut self.orphan_lines);
let state = match states.get_mut(idx) {
Some(Some(draw_state)) => draw_state,
Some(inner) => {
*inner = Some(DrawState::default());
let state = inner.as_mut().unwrap();
state.move_cursor = self.move_cursor;
state.alignment = self.alignment;
state
}
_ => unreachable!(),
};
let member = self.members.get_mut(idx).unwrap();
let state = member.draw_state.get_or_insert(DrawState {
move_cursor: self.move_cursor,
alignment: self.alignment,
..Default::default()
});

DrawStateWrapper::for_multi(state, orphans)
DrawStateWrapper::for_multi(state, &mut self.orphan_lines)
}

pub(crate) fn is_hidden(&self) -> bool {
Expand All @@ -278,12 +309,12 @@ impl MultiState {
fn insert(&mut self, location: InsertLocation) -> usize {
let idx = match self.free_set.pop() {
Some(idx) => {
self.draw_states[idx] = None;
self.members[idx] = MultiStateMember::default();
idx
}
None => {
self.draw_states.push(None);
self.draw_states.len() - 1
self.members.push(MultiStateMember::default());
self.members.len() - 1
}
};

Expand Down Expand Up @@ -328,7 +359,7 @@ impl MultiState {
return;
}

self.draw_states[idx].take();
self.members[idx] = MultiStateMember::default();
self.free_set.push(idx);
self.ordering.retain(|&x| x != idx);

Expand All @@ -340,7 +371,26 @@ impl MultiState {
}

fn len(&self) -> usize {
self.draw_states.len() - self.free_set.len()
self.members.len() - self.free_set.len()
}
}

#[derive(Default)]
struct MultiStateMember {
/// Draw state will be `None` for members that haven't been drawn before, or for entries that
/// correspond to something in the free set.
draw_state: Option<DrawState>,
/// This will be a valid reference unless the containing member is actually in the free set.
pb: Weak<Mutex<BarState>>,
is_zombie: bool,
}

impl Debug for MultiStateMember {
fn fmt(&self, f: &mut Formatter<'_>) -> std::fmt::Result {
f.debug_struct("MultiStateElement")
.field("draw_state", &self.draw_state)
.field("is_zombie", &self.is_zombie)
.finish_non_exhaustive()
}
}

Expand Down Expand Up @@ -422,19 +472,19 @@ mod tests {

let state = mp.state.read().unwrap();
// the removed place for p1 is reused
assert_eq!(state.draw_states.len(), 4);
assert_eq!(state.members.len(), 4);
assert_eq!(state.len(), 3);

// free_set may contain 1 or 2
match state.free_set.last() {
Some(1) => {
assert_eq!(state.ordering, vec![0, 2, 3]);
assert!(state.draw_states[1].is_none());
assert!(state.members[1].draw_state.is_none());
assert_eq!(p4.index().unwrap(), 2);
}
Some(2) => {
assert_eq!(state.ordering, vec![0, 1, 3]);
assert!(state.draw_states[2].is_none());
assert!(state.members[2].draw_state.is_none());
assert_eq!(p4.index().unwrap(), 1);
}
_ => unreachable!(),
Expand Down Expand Up @@ -534,10 +584,10 @@ mod tests {

let state = mp.state.read().unwrap();
// the removed place for p1 is reused
assert_eq!(state.draw_states.len(), 2);
assert_eq!(state.members.len(), 2);
assert_eq!(state.free_set.len(), 1);
assert_eq!(state.len(), 1);
assert!(state.draw_states[0].is_none());
assert!(state.members[0].draw_state.is_none());
assert_eq!(state.free_set.last(), Some(&0));

assert_eq!(state.ordering, vec![1]);
Expand Down
4 changes: 4 additions & 0 deletions src/progress_bar.rs
Original file line number Diff line number Diff line change
Expand Up @@ -266,6 +266,10 @@ impl ProgressBar {
}
}

pub(crate) fn weak_bar_state(&self) -> Weak<Mutex<BarState>> {
Arc::downgrade(&self.state)
}

/// Resets the ETA calculation
///
/// This can be useful if the progress bars made a large jump or was paused for a prolonged
Expand Down
Loading