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

perf(commands): swap shell output as_ref + clone with take #10983

Closed
wants to merge 1 commit into from

Conversation

RoloEdits
Copy link
Contributor

Current as_ref + clone combo is just to deal with lifetime issues of the loop. Instead, take can take ownership directly, without having to clone . This can improve performance for particularly large shell output.

Current `as_ref` + `clone` combo is just to deal with lifetime issues of
the loop. Instead, `take` can take ownership directly, removing
unnecessary operations. This can improve performance for particularly
large shell output
Copy link
Member

@the-mikedavis the-mikedavis left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Replacing the clone with Option::take is the right choice if an option isn't otherwise used but this option is used between iterations of the loop so this is a behavioral change. Instead of reusing the value in shell_output, using take means we consume it (replacing it with None). So this changes the behavior so that we re-run the shell command for every selection rather than reusing the value from the first iteration. That is almost certainly more expensive than cloning the existing value and can lead to a different behavior for some shell usages. For example !echo $RANDOM<ret> with multiple selections sets a random number for every other selection with this branch and only produces one random number on master

@RoloEdits
Copy link
Contributor Author

Hmm, I see. Indeed an interesting change in behavior. Something I might actually play around with more on a branch as that could have its own workflow.

I guess all that could be achieved, as far as performance goes, would be to get rid of a single clone by directly assigning the output_shell with result here? And then do an as_ref().unwrap().clone() every other time.

    let mut output: Option<Tendril> = None;
    let mut offset = 0isize;
    for range in selection.ranges() {
        if output.is_none() {
            let fragment = range.slice(text);
            match shell_impl(shell, cmd, pipe.then(|| fragment.into())) {
                Ok(result) => {
                    output = Some(result);
                }
                Err(err) => {
                    cx.editor.set_error(err.to_string());
                    return;
                }
            }
        };

        let output = output.as_ref().unwrap().clone();
       
      //...

I can pivot to this change, if I hadn't missed anything, or we can just disregard it altogether. #10952 is still open, and this would conflict with that change, so I could wait for that to merge first before pushing here. Its only a single clone saved, and its already small string optimized, most clones could just end up being copies already, so I'll accept whatever direction you want to go.

@the-mikedavis
Copy link
Member

We need to avoid assigning shell_output when pipe is true so that each selection's contents is fed to the shell command. In any case though I don't think it's worth shuffling around this block to avoid a clone per invocation so I will close this out

@RoloEdits RoloEdits deleted the take-shell-output branch June 18, 2024 15:20
Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

2 participants