-
Notifications
You must be signed in to change notification settings - Fork 13k
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
Windows: Don't error on broken non UTF-8 output #134534
Draft
ChrisDenton
wants to merge
3
commits into
rust-lang:master
Choose a base branch
from
ChrisDenton:cp-utf8
base: master
Could not load branches
Branch not found: {{ refName }}
Loading
Could not load tags
Nothing to show
Loading
Are you sure you want to change the base?
Some commits from the old base branch may be removed from the timeline,
and old review comments may become outdated.
Draft
Changes from all commits
Commits
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change | ||||
---|---|---|---|---|---|---|
@@ -1,13 +1,11 @@ | ||||||
#![unstable(issue = "none", feature = "windows_stdio")] | ||||||
|
||||||
use core::str::utf8_char_width; | ||||||
|
||||||
use super::api::{self, WinError}; | ||||||
use crate::mem::MaybeUninit; | ||||||
use crate::os::windows::io::{FromRawHandle, IntoRawHandle}; | ||||||
use crate::sys::handle::Handle; | ||||||
use crate::sys::{c, cvt}; | ||||||
use crate::{cmp, io, ptr, str}; | ||||||
use crate::{cmp, io, ptr}; | ||||||
|
||||||
#[cfg(test)] | ||||||
mod tests; | ||||||
|
@@ -19,13 +17,9 @@ pub struct Stdin { | |||||
incomplete_utf8: IncompleteUtf8, | ||||||
} | ||||||
|
||||||
pub struct Stdout { | ||||||
incomplete_utf8: IncompleteUtf8, | ||||||
} | ||||||
pub struct Stdout {} | ||||||
|
||||||
pub struct Stderr { | ||||||
incomplete_utf8: IncompleteUtf8, | ||||||
} | ||||||
pub struct Stderr {} | ||||||
|
||||||
struct IncompleteUtf8 { | ||||||
bytes: [u8; 4], | ||||||
|
@@ -84,140 +78,69 @@ fn is_console(handle: c::HANDLE) -> bool { | |||||
unsafe { c::GetConsoleMode(handle, &mut mode) != 0 } | ||||||
} | ||||||
|
||||||
fn write(handle_id: u32, data: &[u8], incomplete_utf8: &mut IncompleteUtf8) -> io::Result<usize> { | ||||||
/// Returns true if the attached console's code page is currently UTF-8. | ||||||
#[cfg(not(target_vendor = "win7"))] | ||||||
fn is_utf8_console() -> bool { | ||||||
unsafe { c::GetConsoleOutputCP() == c::CP_UTF8 } | ||||||
} | ||||||
|
||||||
#[cfg(target_vendor = "win7")] | ||||||
fn is_utf8_console() -> bool { | ||||||
// Windows 7 has a fun "feature" where WriteFile on a console handle will return | ||||||
// the number of UTF-16 code units written and not the number of bytes from the input string. | ||||||
// So we always claim the console isn't UTF-8 to trigger the WriteConsole fallback code. | ||||||
false | ||||||
} | ||||||
|
||||||
fn write(handle_id: u32, data: &[u8]) -> io::Result<usize> { | ||||||
if data.is_empty() { | ||||||
return Ok(0); | ||||||
} | ||||||
|
||||||
let handle = get_handle(handle_id)?; | ||||||
if !is_console(handle) { | ||||||
if !is_console(handle) || is_utf8_console() { | ||||||
unsafe { | ||||||
let handle = Handle::from_raw_handle(handle); | ||||||
let ret = handle.write(data); | ||||||
let _ = handle.into_raw_handle(); // Don't close the handle | ||||||
return ret; | ||||||
} | ||||||
} else { | ||||||
write_console_utf16(data, handle) | ||||||
} | ||||||
|
||||||
if incomplete_utf8.len > 0 { | ||||||
assert!( | ||||||
incomplete_utf8.len < 4, | ||||||
"Unexpected number of bytes for incomplete UTF-8 codepoint." | ||||||
); | ||||||
if data[0] >> 6 != 0b10 { | ||||||
// not a continuation byte - reject | ||||||
incomplete_utf8.len = 0; | ||||||
return Err(io::const_error!( | ||||||
io::ErrorKind::InvalidData, | ||||||
"Windows stdio in console mode does not support writing non-UTF-8 byte sequences", | ||||||
)); | ||||||
} | ||||||
incomplete_utf8.bytes[incomplete_utf8.len as usize] = data[0]; | ||||||
incomplete_utf8.len += 1; | ||||||
let char_width = utf8_char_width(incomplete_utf8.bytes[0]); | ||||||
if (incomplete_utf8.len as usize) < char_width { | ||||||
// more bytes needed | ||||||
return Ok(1); | ||||||
} | ||||||
let s = str::from_utf8(&incomplete_utf8.bytes[0..incomplete_utf8.len as usize]); | ||||||
incomplete_utf8.len = 0; | ||||||
match s { | ||||||
Ok(s) => { | ||||||
assert_eq!(char_width, s.len()); | ||||||
let written = write_valid_utf8_to_console(handle, s)?; | ||||||
assert_eq!(written, s.len()); // guaranteed by write_valid_utf8_to_console() for single codepoint writes | ||||||
return Ok(1); | ||||||
} | ||||||
Err(_) => { | ||||||
return Err(io::const_error!( | ||||||
io::ErrorKind::InvalidData, | ||||||
"Windows stdio in console mode does not support writing non-UTF-8 byte sequences", | ||||||
)); | ||||||
} | ||||||
} | ||||||
} | ||||||
|
||||||
// As the console is meant for presenting text, we assume bytes of `data` are encoded as UTF-8, | ||||||
// which needs to be encoded as UTF-16. | ||||||
// | ||||||
// If the data is not valid UTF-8 we write out as many bytes as are valid. | ||||||
// If the first byte is invalid it is either first byte of a multi-byte sequence but the | ||||||
// provided byte slice is too short or it is the first byte of an invalid multi-byte sequence. | ||||||
let len = cmp::min(data.len(), MAX_BUFFER_SIZE / 2); | ||||||
let utf8 = match str::from_utf8(&data[..len]) { | ||||||
Ok(s) => s, | ||||||
Err(ref e) if e.valid_up_to() == 0 => { | ||||||
let first_byte_char_width = utf8_char_width(data[0]); | ||||||
if first_byte_char_width > 1 && data.len() < first_byte_char_width { | ||||||
incomplete_utf8.bytes[0] = data[0]; | ||||||
incomplete_utf8.len = 1; | ||||||
return Ok(1); | ||||||
} else { | ||||||
return Err(io::const_error!( | ||||||
io::ErrorKind::InvalidData, | ||||||
"Windows stdio in console mode does not support writing non-UTF-8 byte sequences", | ||||||
)); | ||||||
} | ||||||
} | ||||||
Err(e) => str::from_utf8(&data[..e.valid_up_to()]).unwrap(), | ||||||
}; | ||||||
|
||||||
write_valid_utf8_to_console(handle, utf8) | ||||||
} | ||||||
|
||||||
fn write_valid_utf8_to_console(handle: c::HANDLE, utf8: &str) -> io::Result<usize> { | ||||||
debug_assert!(!utf8.is_empty()); | ||||||
|
||||||
let mut utf16 = [MaybeUninit::<u16>::uninit(); MAX_BUFFER_SIZE / 2]; | ||||||
let utf8 = &utf8[..utf8.floor_char_boundary(utf16.len())]; | ||||||
fn write_console_utf16(data: &[u8], handle: c::HANDLE) -> io::Result<usize> { | ||||||
let mut buffer = [MaybeUninit::<u16>::uninit(); MAX_BUFFER_SIZE / 2]; | ||||||
let data = &data[..data.len().min(buffer.len())]; | ||||||
|
||||||
// Split off any trailing incomplete UTF-8 from the end of the input. | ||||||
let utf8 = trim_last_char_boundary(data); | ||||||
let utf16 = utf8_to_utf16_lossy(utf8, &mut buffer); | ||||||
debug_assert!(!utf16.is_empty()); | ||||||
|
||||||
// Write the UTF-16 chars to the console. | ||||||
// This will succeed in one write so long as our [u16] slice is smaller than the console's buffer, | ||||||
// which we've ensured by truncating the input (see `MAX_BUFFER_SIZE`). | ||||||
let written = write_u16s(handle, &utf16)?; | ||||||
debug_assert_eq!(written, utf16.len()); | ||||||
Ok(utf8.len()) | ||||||
} | ||||||
|
||||||
let utf16: &[u16] = unsafe { | ||||||
// Note that this theoretically checks validity twice in the (most common) case | ||||||
// where the underlying byte sequence is valid utf-8 (given the check in `write()`). | ||||||
fn utf8_to_utf16_lossy<'a>(utf8: &[u8], utf16: &'a mut [MaybeUninit<u16>]) -> &'a [u16] { | ||||||
unsafe { | ||||||
let result = c::MultiByteToWideChar( | ||||||
c::CP_UTF8, // CodePage | ||||||
c::MB_ERR_INVALID_CHARS, // dwFlags | ||||||
0, // dwFlags | ||||||
utf8.as_ptr(), // lpMultiByteStr | ||||||
utf8.len() as i32, // cbMultiByte | ||||||
utf16.as_mut_ptr() as *mut c::WCHAR, // lpWideCharStr | ||||||
utf16.len() as i32, // cchWideChar | ||||||
); | ||||||
assert!(result != 0, "Unexpected error in MultiByteToWideChar"); | ||||||
|
||||||
// The only way an error can happen here is if we've messed up. | ||||||
debug_assert!(result != 0, "Unexpected error in MultiByteToWideChar"); | ||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
I think this should be an |
||||||
// Safety: MultiByteToWideChar initializes `result` values. | ||||||
MaybeUninit::slice_assume_init_ref(&utf16[..result as usize]) | ||||||
}; | ||||||
|
||||||
let mut written = write_u16s(handle, utf16)?; | ||||||
|
||||||
// Figure out how many bytes of as UTF-8 were written away as UTF-16. | ||||||
if written == utf16.len() { | ||||||
Ok(utf8.len()) | ||||||
} else { | ||||||
// Make sure we didn't end up writing only half of a surrogate pair (even though the chance | ||||||
// is tiny). Because it is not possible for user code to re-slice `data` in such a way that | ||||||
// a missing surrogate can be produced (and also because of the UTF-8 validation above), | ||||||
// write the missing surrogate out now. | ||||||
// Buffering it would mean we have to lie about the number of bytes written. | ||||||
let first_code_unit_remaining = utf16[written]; | ||||||
if matches!(first_code_unit_remaining, 0xDCEE..=0xDFFF) { | ||||||
// low surrogate | ||||||
// We just hope this works, and give up otherwise | ||||||
let _ = write_u16s(handle, &utf16[written..written + 1]); | ||||||
written += 1; | ||||||
} | ||||||
// Calculate the number of bytes of `utf8` that were actually written. | ||||||
let mut count = 0; | ||||||
for ch in utf16[..written].iter() { | ||||||
count += match ch { | ||||||
0x0000..=0x007F => 1, | ||||||
0x0080..=0x07FF => 2, | ||||||
0xDCEE..=0xDFFF => 1, // Low surrogate. We already counted 3 bytes for the other. | ||||||
_ => 3, | ||||||
}; | ||||||
} | ||||||
debug_assert!(String::from_utf16(&utf16[..written]).unwrap() == utf8[..count]); | ||||||
Ok(count) | ||||||
} | ||||||
} | ||||||
|
||||||
|
@@ -410,13 +333,13 @@ impl IncompleteUtf8 { | |||||
|
||||||
impl Stdout { | ||||||
pub const fn new() -> Stdout { | ||||||
Stdout { incomplete_utf8: IncompleteUtf8::new() } | ||||||
Stdout {} | ||||||
} | ||||||
} | ||||||
|
||||||
impl io::Write for Stdout { | ||||||
fn write(&mut self, buf: &[u8]) -> io::Result<usize> { | ||||||
write(c::STD_OUTPUT_HANDLE, buf, &mut self.incomplete_utf8) | ||||||
write(c::STD_OUTPUT_HANDLE, buf) | ||||||
} | ||||||
|
||||||
fn flush(&mut self) -> io::Result<()> { | ||||||
|
@@ -426,13 +349,13 @@ impl io::Write for Stdout { | |||||
|
||||||
impl Stderr { | ||||||
pub const fn new() -> Stderr { | ||||||
Stderr { incomplete_utf8: IncompleteUtf8::new() } | ||||||
Stderr {} | ||||||
} | ||||||
} | ||||||
|
||||||
impl io::Write for Stderr { | ||||||
fn write(&mut self, buf: &[u8]) -> io::Result<usize> { | ||||||
write(c::STD_ERROR_HANDLE, buf, &mut self.incomplete_utf8) | ||||||
write(c::STD_ERROR_HANDLE, buf) | ||||||
} | ||||||
|
||||||
fn flush(&mut self) -> io::Result<()> { | ||||||
|
@@ -447,3 +370,50 @@ pub fn is_ebadf(err: &io::Error) -> bool { | |||||
pub fn panic_output() -> Option<impl io::Write> { | ||||||
Some(Stderr::new()) | ||||||
} | ||||||
|
||||||
/// Trim one incomplete UTF-8 char from the end of a byte slice. | ||||||
/// | ||||||
/// If trimming would lead to an empty slice then it returns `bytes` instead. | ||||||
/// | ||||||
/// Note: This function is optimized for size rather than speed. | ||||||
pub fn trim_last_char_boundary(bytes: &[u8]) -> &[u8] { | ||||||
// UTF-8's multiple-byte encoding uses the leading bits to encode the length of a code point. | ||||||
// The bits of a multi-byte sequence are (where `n` is a placeholder for any bit): | ||||||
// | ||||||
// 11110nnn 10nnnnnn 10nnnnnn 10nnnnnn | ||||||
// 1110nnnn 10nnnnnn 10nnnnnn | ||||||
// 110nnnnn 10nnnnnn | ||||||
// | ||||||
// So if follows that an incomplete sequence is one of these: | ||||||
// 11110nnn 10nnnnnn 10nnnnnn | ||||||
// 11110nnn 10nnnnnn | ||||||
// 1110nnnn 10nnnnnn | ||||||
// 11110nnn | ||||||
// 1110nnnn | ||||||
// 110nnnnn | ||||||
|
||||||
// Get up to three bytes from the end of the slice and encode them as a u32 | ||||||
// because it turns out the compiler is very good at optimizing numbers. | ||||||
let u = match bytes { | ||||||
[.., b1, b2, b3] => (*b1 as u32) << 16 | (*b2 as u32) << 8 | *b3 as u32, | ||||||
[.., b1, b2] => (*b1 as u32) << 8 | *b2 as u32, | ||||||
// If it's just a single byte or empty then we return the full slice | ||||||
_ => return bytes, | ||||||
}; | ||||||
if (u & 0b_11111000_11000000_11000000 == 0b_11110000_10000000_10000000) && bytes.len() >= 4 { | ||||||
&bytes[..bytes.len() - 3] | ||||||
} else if (u & 0b_11111000_11000000 == 0b_11110000_10000000 | ||||||
|| u & 0b_11110000_11000000 == 0b_11100000_10000000) | ||||||
&& bytes.len() >= 3 | ||||||
{ | ||||||
&bytes[..bytes.len() - 2] | ||||||
} else if (u & 0b_1111_1000 == 0b_1111_0000 | ||||||
|| u & 0b_11110000 == 0b_11100000 | ||||||
|| u & 0b_11100000 == 0b_11000000) | ||||||
&& bytes.len() >= 2 | ||||||
{ | ||||||
&bytes[..bytes.len() - 1] | ||||||
} else { | ||||||
bytes | ||||||
} | ||||||
} |
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The reason for the removal of incomplete UTF-8 handling at the end of the string is not clear to me from the commit description. Why was that removed?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Because it now simply truncates the write to remove incomplete UTF-8 from the end and instead leaves the buffering to buffer types, i.e.
LineWriter
in this case.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
A
LineWriter
will flush an incomplete line if its buffer capacity is exceeded. If that happens, the output must support partial UTF-8 writes, or non-ASCII characters might get lost or replaced with the replacement character.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That can only result in broken UTF-8 if the user writes incomplete UTF-8 to
LineWriter
themselves.There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I see, digging through the source code,
BufWriter
makes sure to not split writes that the user issued.What is the motivation for truncating invalid UTF-8 at the end of the string?
All else being equal, I'd rather expect the previous behavior, that I can construct UTF-8 output byte-by-byte.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Rather than having a secret stack buffer that can't be inspected or flushed, I'd strongly prefer buffering be done at a higher level. It's also a lot of added complexity for an edge case where the better solution is to set the console code page.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
In any case, if that behavior is wanted, it should probably be documented in the commit message so that it is clear to future readers that this change was on purpose.
Rather than ignoring trailing invalid UTF-8, I think it'd be better to replace it with a replacement character so that it becomes clear that something was removed.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
That's what happens in this code. No bytes are ever lost. Either the caller is informed that less bytes were written than were provided or, if there is only an incomplete code point, then that is written to the console (which will be converted to replacement characters when lossy translating to UTF-16).
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Ah yes, that makes sense. I didn't realize the caller would be informed by the return value of
write
.What is the motivation for special casing trailing invalid UTF-8? It seems to increase the code complexity a little as well, and is not necessary for std's own use cases.
Is it for supporting a potential non-std buffered writer?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Sure, it could be removed.
Stderr
is not buffered by us though and there have been proposals for unbuffered stdout.