diff options
| author | Oliver Middleton <olliemail27@gmail.com> | 2016-03-10 19:09:02 +0000 |
|---|---|---|
| committer | Oliver Middleton <olliemail27@gmail.com> | 2016-03-10 21:15:23 +0000 |
| commit | 8427efaab377346c0c4be73664422897a4072330 (patch) | |
| tree | 69723a049b19d200f1eb35ff9778750f894282d6 /src/libstd/sys | |
| parent | 4b87655e69a6d07f03b104caaec80a042ad40bf1 (diff) | |
| download | rust-8427efaab377346c0c4be73664422897a4072330.tar.gz rust-8427efaab377346c0c4be73664422897a4072330.zip | |
Fixup stout/stderr on Windows
WriteConsoleW can fail if called with a large buffer so we need to slice any stdout/stderr output. However the current slicing has a few problems: 1. It slices by byte but still expects valid UTF-8. 2. The slicing happens even when not outputting to a console. 3. panic! output is not sliced. This fixes these issues by moving the slice to right before WriteConsoleW and slicing on a char boundary.
Diffstat (limited to 'src/libstd/sys')
| -rw-r--r-- | src/libstd/sys/windows/stdio.rs | 26 |
1 files changed, 24 insertions, 2 deletions
diff --git a/src/libstd/sys/windows/stdio.rs b/src/libstd/sys/windows/stdio.rs index 5883904c21d..190c2571628 100644 --- a/src/libstd/sys/windows/stdio.rs +++ b/src/libstd/sys/windows/stdio.rs @@ -58,8 +58,30 @@ fn write(out: &Output, data: &[u8]) -> io::Result<usize> { Output::Console(ref c) => c.get().raw(), Output::Pipe(ref p) => return p.get().write(data), }; + // As with stdin on windows, stdout often can't handle writes of large + // sizes. For an example, see #14940. For this reason, don't try to + // write the entire output buffer on windows. + // + // For some other references, it appears that this problem has been + // encountered by others [1] [2]. We choose the number 8K just because + // libuv does the same. + // + // [1]: https://tahoe-lafs.org/trac/tahoe-lafs/ticket/1232 + // [2]: http://www.mail-archive.com/log4net-dev@logging.apache.org/msg00661.html + const OUT_MAX: usize = 8192; + let data_len; let utf16 = match str::from_utf8(data).ok() { - Some(utf8) => utf8.encode_utf16().collect::<Vec<u16>>(), + Some(mut utf8) => { + if utf8.len() > OUT_MAX { + let mut new_len = OUT_MAX; + while !utf8.is_char_boundary(new_len) { + new_len -= 1; + } + utf8 = &utf8[..new_len]; + } + data_len = utf8.len(); + utf8.encode_utf16().collect::<Vec<u16>>() + } None => return Err(invalid_encoding()), }; let mut written = 0; @@ -74,7 +96,7 @@ fn write(out: &Output, data: &[u8]) -> io::Result<usize> { // FIXME if this only partially writes the utf16 buffer then we need to // figure out how many bytes of `data` were actually written assert_eq!(written as usize, utf16.len()); - Ok(data.len()) + Ok(data_len) } impl Stdin { |
