From e90408e91596da9fd81e77012e74bfe94b26dd83 Mon Sep 17 00:00:00 2001 From: Brad King Date: Sat, 25 Apr 2015 15:35:22 -0400 Subject: [PATCH] std: Fix process spawn for arguments ending in backslashes on Windows Fix `make_command_line` for the case of backslashes at the end of an argument requiring quotes. We must encode the command and arguments such that `CommandLineToArgvW` recovers them in the spawned process. Simplify the logic by using a running count of backslashes as they are encountered instead of looking ahead for quotes following them. Extend `test_make_command_line` to additionally cover: * a leading quote in an argument that requires quotes, * a backslash before a quote in an argument that requires quotes, * a backslash at the end of an argument that requires quotes, and * a backslash at the end of an argument that does not require quotes. --- src/libstd/sys/windows/process2.rs | 39 ++++++++++++++++-------------- 1 file changed, 21 insertions(+), 18 deletions(-) diff --git a/src/libstd/sys/windows/process2.rs b/src/libstd/sys/windows/process2.rs index 5ddcf3d1ea299..2e5585d2f4389 100644 --- a/src/libstd/sys/windows/process2.rs +++ b/src/libstd/sys/windows/process2.rs @@ -367,6 +367,8 @@ fn zeroed_process_information() -> libc::types::os::arch::extra::PROCESS_INFORMA // Produces a wide string *without terminating null* fn make_command_line(prog: &OsStr, args: &[OsString]) -> Vec { + // Encode the command and arguments in a command line string such + // that the spawned process may recover them using CommandLineToArgvW. let mut cmd: Vec = Vec::new(); append_arg(&mut cmd, prog); for arg in args { @@ -387,30 +389,27 @@ fn make_command_line(prog: &OsStr, args: &[OsString]) -> Vec { } let mut iter = arg.encode_wide(); + let mut backslashes: usize = 0; while let Some(x) = iter.next() { - if x == '"' as u16 { - // escape quotes - cmd.push('\\' as u16); - cmd.push('"' as u16); - } else if x == '\\' as u16 { - // is this a run of backslashes followed by a " ? - if iter.clone().skip_while(|y| *y == '\\' as u16).next() == Some('"' as u16) { - // Double it ... NOTE: this behavior is being - // preserved as it's been part of Rust for a long - // time, but no one seems to know exactly why this - // is the right thing to do. - cmd.push('\\' as u16); - cmd.push('\\' as u16); - } else { - // Push it through unescaped - cmd.push('\\' as u16); - } + if x == '\\' as u16 { + backslashes += 1; } else { - cmd.push(x) + if x == '"' as u16 { + // Add n+1 backslashes to total 2n+1 before internal '"'. + for _ in 0..(backslashes+1) { + cmd.push('\\' as u16); + } + } + backslashes = 0; } + cmd.push(x); } if quote { + // Add n backslashes to total 2n before ending '"'. + for _ in 0..backslashes { + cmd.push('\\' as u16); + } cmd.push('"' as u16); } } @@ -486,6 +485,10 @@ mod tests { test_wrapper("echo", &["a b c"]), "echo \"a b c\"" ); + assert_eq!( + test_wrapper("echo", &["\" \\\" \\", "\\"]), + "echo \"\\\" \\\\\\\" \\\\\" \\" + ); assert_eq!( test_wrapper("\u{03c0}\u{042f}\u{97f3}\u{00e6}\u{221e}", &[]), "\u{03c0}\u{042f}\u{97f3}\u{00e6}\u{221e}"